#ifndef TSF_INCLUDE_TSF_INL
#define TSF_INCLUDE_TSF_INL
#ifdef __cplusplus
extern "C" {
# define CPP_DEFAULT0 = 0
#else
# define CPP_DEFAULT0
#endif
#ifdef TSF_STATIC
#define TSFDEF static
#else
#define TSFDEF extern
#endif
typedef struct tsf tsf;
#ifndef TSF_NO_STDIO
TSFDEF tsf* tsf_load_filename(const char* filename);
#endif
TSFDEF tsf* tsf_load_memory(const void* buffer, int size);
struct tsf_stream
{
void* data;
int (*read)(void* data, void* ptr, unsigned int size);
int (*skip)(void* data, unsigned int count);
};
TSFDEF tsf* tsf_load(struct tsf_stream* stream);
TSFDEF void tsf_close(tsf* f);
TSFDEF void tsf_reset(tsf* f);
TSFDEF int tsf_get_presetindex(const tsf* f, int bank, int preset_number);
TSFDEF int tsf_get_presetcount(const tsf* f);
TSFDEF const char* tsf_get_presetname(const tsf* f, int preset_index);
TSFDEF const char* tsf_bank_get_presetname(const tsf* f, int bank, int preset_number);
enum TSFOutputMode
{
TSF_STEREO_INTERLEAVED,
TSF_STEREO_UNWEAVED,
TSF_MONO,
};
TSFDEF void tsf_set_output(tsf* f, enum TSFOutputMode outputmode, int samplerate, float global_gain_db CPP_DEFAULT0);
TSFDEF void tsf_set_volume(tsf* f, float global_gain);
TSFDEF void tsf_set_max_voices(tsf* f, int max_voices);
TSFDEF void tsf_note_on(tsf* f, int preset_index, int key, float vel);
TSFDEF int tsf_bank_note_on(tsf* f, int bank, int preset_number, int key, float vel);
TSFDEF void tsf_note_off(tsf* f, int preset_index, int key);
TSFDEF int tsf_bank_note_off(tsf* f, int bank, int preset_number, int key);
TSFDEF void tsf_note_off_all(tsf* f);
TSFDEF int tsf_active_voice_count(tsf* f);
TSFDEF void tsf_render_short(tsf* f, short* buffer, int samples, int flag_mixing CPP_DEFAULT0);
TSFDEF void tsf_render_float(tsf* f, float* buffer, int samples, int flag_mixing CPP_DEFAULT0);
TSFDEF void tsf_channel_set_presetindex(tsf* f, int channel, int preset_index);
TSFDEF int tsf_channel_set_presetnumber(tsf* f, int channel, int preset_number, int flag_mididrums CPP_DEFAULT0);
TSFDEF void tsf_channel_set_bank(tsf* f, int channel, int bank);
TSFDEF int tsf_channel_set_bank_preset(tsf* f, int channel, int bank, int preset_number);
TSFDEF void tsf_channel_set_pan(tsf* f, int channel, float pan);
TSFDEF void tsf_channel_set_volume(tsf* f, int channel, float volume);
TSFDEF void tsf_channel_set_pitchwheel(tsf* f, int channel, int pitch_wheel);
TSFDEF void tsf_channel_set_pitchrange(tsf* f, int channel, float pitch_range);
TSFDEF void tsf_channel_set_tuning(tsf* f, int channel, float tuning);
TSFDEF void tsf_channel_note_on(tsf* f, int channel, int key, float vel);
TSFDEF void tsf_channel_note_off(tsf* f, int channel, int key);
TSFDEF void tsf_channel_note_off_all(tsf* f, int channel); TSFDEF void tsf_channel_sounds_off_all(tsf* f, int channel);
TSFDEF void tsf_channel_midi_control(tsf* f, int channel, int controller, int control_value);
TSFDEF int tsf_channel_get_preset_index(tsf* f, int channel);
TSFDEF int tsf_channel_get_preset_bank(tsf* f, int channel);
TSFDEF int tsf_channel_get_preset_number(tsf* f, int channel);
TSFDEF float tsf_channel_get_pan(tsf* f, int channel);
TSFDEF float tsf_channel_get_volume(tsf* f, int channel);
TSFDEF int tsf_channel_get_pitchwheel(tsf* f, int channel);
TSFDEF float tsf_channel_get_pitchrange(tsf* f, int channel);
TSFDEF float tsf_channel_get_tuning(tsf* f, int channel);
#ifdef __cplusplus
# undef CPP_DEFAULT0
}
#endif
#endif
#ifdef TSF_IMPLEMENTATION
#undef TSF_IMPLEMENTATION
#ifndef TSF_RENDER_EFFECTSAMPLEBLOCK
#define TSF_RENDER_EFFECTSAMPLEBLOCK 64
#endif
#define TSF_FASTRELEASETIME 0.01f
#if !defined(TSF_MALLOC) || !defined(TSF_FREE) || !defined(TSF_REALLOC)
# include <stdlib.h>
# define TSF_MALLOC malloc
# define TSF_FREE free
# define TSF_REALLOC realloc
#endif
#if !defined(TSF_MEMCPY) || !defined(TSF_MEMSET)
# include <string.h>
# define TSF_MEMCPY memcpy
# define TSF_MEMSET memset
#endif
#if !defined(TSF_POW) || !defined(TSF_POWF) || !defined(TSF_EXPF) || !defined(TSF_LOG) || !defined(TSF_TAN) || !defined(TSF_LOG10) || !defined(TSF_SQRT)
# include <math.h>
# if !defined(__cplusplus) && !defined(NAN) && !defined(powf) && !defined(expf) && !defined(sqrtf)
# define powf (float)pow
# define expf (float)exp
# define sqrtf (float)sqrt
# endif
# define TSF_POW pow
# define TSF_POWF powf
# define TSF_EXPF expf
# define TSF_LOG log
# define TSF_TAN tan
# define TSF_LOG10 log10
# define TSF_SQRTF sqrtf
#endif
#ifndef TSF_NO_STDIO
# include <stdio.h>
#endif
#define TSF_TRUE 1
#define TSF_FALSE 0
#define TSF_BOOL char
#define TSF_PI 3.14159265358979323846264338327950288
#define TSF_NULL 0
#ifdef __cplusplus
extern "C" {
#endif
typedef char tsf_fourcc[4];
typedef signed char tsf_s8;
typedef unsigned char tsf_u8;
typedef unsigned short tsf_u16;
typedef signed short tsf_s16;
typedef unsigned int tsf_u32;
typedef char tsf_char20[20];
#define TSF_FourCCEquals(value1, value2) (value1[0] == value2[0] && value1[1] == value2[1] && value1[2] == value2[2] && value1[3] == value2[3])
struct tsf
{
struct tsf_preset* presets;
float* fontSamples;
struct tsf_voice* voices;
struct tsf_channels* channels;
float* outputSamples;
int presetNum;
int voiceNum;
int maxVoiceNum;
int outputSampleSize;
unsigned int voicePlayIndex;
enum TSFOutputMode outputmode;
float outSampleRate;
float globalGainDB;
};
#ifndef TSF_NO_STDIO
static int tsf_stream_stdio_read(FILE* f, void* ptr, unsigned int size) { return (int)fread(ptr, 1, size, f); }
static int tsf_stream_stdio_skip(FILE* f, unsigned int count) { return !fseek(f, count, SEEK_CUR); }
TSFDEF tsf* tsf_load_filename(const char* filename)
{
tsf* res;
struct tsf_stream stream = { TSF_NULL, (int(*)(void*,void*,unsigned int))&tsf_stream_stdio_read, (int(*)(void*,unsigned int))&tsf_stream_stdio_skip };
#if __STDC_WANT_SECURE_LIB__
FILE* f = TSF_NULL; fopen_s(&f, filename, "rb");
#else
FILE* f = fopen(filename, "rb");
#endif
if (!f)
{
return TSF_NULL;
}
stream.data = f;
res = tsf_load(&stream);
fclose(f);
return res;
}
#endif
struct tsf_stream_memory { const char* buffer; unsigned int total, pos; };
static int tsf_stream_memory_read(struct tsf_stream_memory* m, void* ptr, unsigned int size) { if (size > m->total - m->pos) size = m->total - m->pos; TSF_MEMCPY(ptr, m->buffer+m->pos, size); m->pos += size; return size; }
static int tsf_stream_memory_skip(struct tsf_stream_memory* m, unsigned int count) { if (m->pos + count > m->total) return 0; m->pos += count; return 1; }
TSFDEF tsf* tsf_load_memory(const void* buffer, int size)
{
struct tsf_stream stream = { TSF_NULL, (int(*)(void*,void*,unsigned int))&tsf_stream_memory_read, (int(*)(void*,unsigned int))&tsf_stream_memory_skip };
struct tsf_stream_memory f = { 0, 0, 0 };
f.buffer = (const char*)buffer;
f.total = size;
stream.data = &f;
return tsf_load(&stream);
}
enum { TSF_LOOPMODE_NONE, TSF_LOOPMODE_CONTINUOUS, TSF_LOOPMODE_SUSTAIN };
enum { TSF_SEGMENT_NONE, TSF_SEGMENT_DELAY, TSF_SEGMENT_ATTACK, TSF_SEGMENT_HOLD, TSF_SEGMENT_DECAY, TSF_SEGMENT_SUSTAIN, TSF_SEGMENT_RELEASE, TSF_SEGMENT_DONE };
struct tsf_hydra
{
struct tsf_hydra_phdr *phdrs; struct tsf_hydra_pbag *pbags; struct tsf_hydra_pmod *pmods;
struct tsf_hydra_pgen *pgens; struct tsf_hydra_inst *insts; struct tsf_hydra_ibag *ibags;
struct tsf_hydra_imod *imods; struct tsf_hydra_igen *igens; struct tsf_hydra_shdr *shdrs;
int phdrNum, pbagNum, pmodNum, pgenNum, instNum, ibagNum, imodNum, igenNum, shdrNum;
};
union tsf_hydra_genamount { struct { tsf_u8 lo, hi; } range; tsf_s16 shortAmount; tsf_u16 wordAmount; };
struct tsf_hydra_phdr { tsf_char20 presetName; tsf_u16 preset, bank, presetBagNdx; tsf_u32 library, genre, morphology; };
struct tsf_hydra_pbag { tsf_u16 genNdx, modNdx; };
struct tsf_hydra_pmod { tsf_u16 modSrcOper, modDestOper; tsf_s16 modAmount; tsf_u16 modAmtSrcOper, modTransOper; };
struct tsf_hydra_pgen { tsf_u16 genOper; union tsf_hydra_genamount genAmount; };
struct tsf_hydra_inst { tsf_char20 instName; tsf_u16 instBagNdx; };
struct tsf_hydra_ibag { tsf_u16 instGenNdx, instModNdx; };
struct tsf_hydra_imod { tsf_u16 modSrcOper, modDestOper; tsf_s16 modAmount; tsf_u16 modAmtSrcOper, modTransOper; };
struct tsf_hydra_igen { tsf_u16 genOper; union tsf_hydra_genamount genAmount; };
struct tsf_hydra_shdr { tsf_char20 sampleName; tsf_u32 start, end, startLoop, endLoop, sampleRate; tsf_u8 originalPitch; tsf_s8 pitchCorrection; tsf_u16 sampleLink, sampleType; };
#define TSFR(FIELD) stream->read(stream->data, &i->FIELD, sizeof(i->FIELD));
static void tsf_hydra_read_phdr(struct tsf_hydra_phdr* i, struct tsf_stream* stream) { TSFR(presetName) TSFR(preset) TSFR(bank) TSFR(presetBagNdx) TSFR(library) TSFR(genre) TSFR(morphology) }
static void tsf_hydra_read_pbag(struct tsf_hydra_pbag* i, struct tsf_stream* stream) { TSFR(genNdx) TSFR(modNdx) }
static void tsf_hydra_read_pmod(struct tsf_hydra_pmod* i, struct tsf_stream* stream) { TSFR(modSrcOper) TSFR(modDestOper) TSFR(modAmount) TSFR(modAmtSrcOper) TSFR(modTransOper) }
static void tsf_hydra_read_pgen(struct tsf_hydra_pgen* i, struct tsf_stream* stream) { TSFR(genOper) TSFR(genAmount) }
static void tsf_hydra_read_inst(struct tsf_hydra_inst* i, struct tsf_stream* stream) { TSFR(instName) TSFR(instBagNdx) }
static void tsf_hydra_read_ibag(struct tsf_hydra_ibag* i, struct tsf_stream* stream) { TSFR(instGenNdx) TSFR(instModNdx) }
static void tsf_hydra_read_imod(struct tsf_hydra_imod* i, struct tsf_stream* stream) { TSFR(modSrcOper) TSFR(modDestOper) TSFR(modAmount) TSFR(modAmtSrcOper) TSFR(modTransOper) }
static void tsf_hydra_read_igen(struct tsf_hydra_igen* i, struct tsf_stream* stream) { TSFR(genOper) TSFR(genAmount) }
static void tsf_hydra_read_shdr(struct tsf_hydra_shdr* i, struct tsf_stream* stream) { TSFR(sampleName) TSFR(start) TSFR(end) TSFR(startLoop) TSFR(endLoop) TSFR(sampleRate) TSFR(originalPitch) TSFR(pitchCorrection) TSFR(sampleLink) TSFR(sampleType) }
#undef TSFR
struct tsf_riffchunk { tsf_fourcc id; tsf_u32 size; };
struct tsf_envelope { float delay, attack, hold, decay, sustain, release, keynumToHold, keynumToDecay; };
struct tsf_voice_envelope { float level, slope; int samplesUntilNextSegment; short segment, midiVelocity; struct tsf_envelope parameters; TSF_BOOL segmentIsExponential, isAmpEnv; };
struct tsf_voice_lowpass { double QInv, a0, a1, b1, b2, z1, z2; TSF_BOOL active; };
struct tsf_voice_lfo { int samplesUntil; float level, delta; };
struct tsf_region
{
int loop_mode;
unsigned int sample_rate;
unsigned char lokey, hikey, lovel, hivel;
unsigned int group, offset, end, loop_start, loop_end;
int transpose, tune, pitch_keycenter, pitch_keytrack;
float attenuation, pan;
struct tsf_envelope ampenv, modenv;
int initialFilterQ, initialFilterFc;
int modEnvToPitch, modEnvToFilterFc, modLfoToFilterFc, modLfoToVolume;
float delayModLFO;
int freqModLFO, modLfoToPitch;
float delayVibLFO;
int freqVibLFO, vibLfoToPitch;
};
struct tsf_preset
{
tsf_char20 presetName;
tsf_u16 preset, bank;
struct tsf_region* regions;
int regionNum;
};
struct tsf_voice
{
int playingPreset, playingKey, playingChannel;
struct tsf_region* region;
double pitchInputTimecents, pitchOutputFactor;
double sourceSamplePosition;
float noteGainDB, panFactorLeft, panFactorRight;
unsigned int playIndex, loopStart, loopEnd;
struct tsf_voice_envelope ampenv, modenv;
struct tsf_voice_lowpass lowpass;
struct tsf_voice_lfo modlfo, viblfo;
};
struct tsf_channel
{
unsigned short presetIndex, bank, pitchWheel, midiPan, midiVolume, midiExpression, midiRPN, midiData;
float panOffset, gainDB, pitchRange, tuning;
};
struct tsf_channels
{
void (*setupVoice)(tsf* f, struct tsf_voice* voice);
struct tsf_channel* channels;
int channelNum, activeChannel;
};
static double tsf_timecents2Secsd(double timecents) { return TSF_POW(2.0, timecents / 1200.0); }
static float tsf_timecents2Secsf(float timecents) { return TSF_POWF(2.0f, timecents / 1200.0f); }
static float tsf_cents2Hertz(float cents) { return 8.176f * TSF_POWF(2.0f, cents / 1200.0f); }
static float tsf_decibelsToGain(float db) { return (db > -100.f ? TSF_POWF(10.0f, db * 0.05f) : 0); }
static float tsf_gainToDecibels(float gain) { return (gain <= .00001f ? -100.f : (float)(20.0 * TSF_LOG10(gain))); }
static TSF_BOOL tsf_riffchunk_read(struct tsf_riffchunk* parent, struct tsf_riffchunk* chunk, struct tsf_stream* stream)
{
TSF_BOOL IsRiff, IsList;
if (parent && sizeof(tsf_fourcc) + sizeof(tsf_u32) > parent->size) return TSF_FALSE;
if (!stream->read(stream->data, &chunk->id, sizeof(tsf_fourcc)) || *chunk->id <= ' ' || *chunk->id >= 'z') return TSF_FALSE;
if (!stream->read(stream->data, &chunk->size, sizeof(tsf_u32))) return TSF_FALSE;
if (parent && sizeof(tsf_fourcc) + sizeof(tsf_u32) + chunk->size > parent->size) return TSF_FALSE;
if (parent) parent->size -= sizeof(tsf_fourcc) + sizeof(tsf_u32) + chunk->size;
IsRiff = TSF_FourCCEquals(chunk->id, "RIFF"), IsList = TSF_FourCCEquals(chunk->id, "LIST");
if (IsRiff && parent) return TSF_FALSE; if (!IsRiff && !IsList) return TSF_TRUE; if (!stream->read(stream->data, &chunk->id, sizeof(tsf_fourcc)) || *chunk->id <= ' ' || *chunk->id >= 'z') return TSF_FALSE;
chunk->size -= sizeof(tsf_fourcc);
return TSF_TRUE;
}
static void tsf_region_clear(struct tsf_region* i, TSF_BOOL for_relative)
{
TSF_MEMSET(i, 0, sizeof(struct tsf_region));
i->hikey = i->hivel = 127;
i->pitch_keycenter = 60; if (for_relative) return;
i->pitch_keytrack = 100;
i->pitch_keycenter = -1;
i->ampenv.delay = i->ampenv.attack = i->ampenv.hold = i->ampenv.decay = i->ampenv.release = -12000.0f;
i->modenv.delay = i->modenv.attack = i->modenv.hold = i->modenv.decay = i->modenv.release = -12000.0f;
i->initialFilterFc = 13500;
i->delayModLFO = -12000.0f;
i->delayVibLFO = -12000.0f;
}
static void tsf_region_operator(struct tsf_region* region, tsf_u16 genOper, union tsf_hydra_genamount* amount, struct tsf_region* merge_region)
{
enum
{
_GEN_TYPE_MASK = 0x0F,
GEN_FLOAT = 0x01,
GEN_INT = 0x02,
GEN_UINT_ADD = 0x03,
GEN_UINT_ADD15 = 0x04,
GEN_KEYRANGE = 0x05,
GEN_VELRANGE = 0x06,
GEN_LOOPMODE = 0x07,
GEN_GROUP = 0x08,
GEN_KEYCENTER = 0x09,
_GEN_LIMIT_MASK = 0xF0,
GEN_INT_LIMIT12K = 0x10, GEN_INT_LIMITFC = 0x20, GEN_INT_LIMITQ = 0x30, GEN_INT_LIMIT960 = 0x40, GEN_INT_LIMIT16K4500 = 0x50, GEN_FLOAT_LIMIT12K5K = 0x60, GEN_FLOAT_LIMIT12K8K = 0x70, GEN_FLOAT_LIMIT1200 = 0x80, GEN_FLOAT_LIMITPAN = 0x90, GEN_FLOAT_LIMITATTN = 0xA0, GEN_FLOAT_MAX1000 = 0xB0, GEN_FLOAT_MAX1440 = 0xC0,
_GEN_MAX = 59,
};
#define _TSFREGIONOFFSET(TYPE, FIELD) (unsigned char)(((TYPE*)&((struct tsf_region*)0)->FIELD) - (TYPE*)0)
#define _TSFREGIONENVOFFSET(TYPE, ENV, FIELD) (unsigned char)(((TYPE*)&((&(((struct tsf_region*)0)->ENV))->FIELD)) - (TYPE*)0)
static const struct { unsigned char mode, offset; } genMetas[_GEN_MAX] =
{
{ GEN_UINT_ADD , _TSFREGIONOFFSET(unsigned int, offset ) }, { GEN_UINT_ADD , _TSFREGIONOFFSET(unsigned int, end ) }, { GEN_UINT_ADD , _TSFREGIONOFFSET(unsigned int, loop_start ) }, { GEN_UINT_ADD , _TSFREGIONOFFSET(unsigned int, loop_end ) }, { GEN_UINT_ADD15 , _TSFREGIONOFFSET(unsigned int, offset ) }, { GEN_INT | GEN_INT_LIMIT12K , _TSFREGIONOFFSET( int, modLfoToPitch ) }, { GEN_INT | GEN_INT_LIMIT12K , _TSFREGIONOFFSET( int, vibLfoToPitch ) }, { GEN_INT | GEN_INT_LIMIT12K , _TSFREGIONOFFSET( int, modEnvToPitch ) }, { GEN_INT | GEN_INT_LIMITFC , _TSFREGIONOFFSET( int, initialFilterFc ) }, { GEN_INT | GEN_INT_LIMITQ , _TSFREGIONOFFSET( int, initialFilterQ ) }, { GEN_INT | GEN_INT_LIMIT12K , _TSFREGIONOFFSET( int, modLfoToFilterFc ) }, { GEN_INT | GEN_INT_LIMIT12K , _TSFREGIONOFFSET( int, modEnvToFilterFc ) }, { GEN_UINT_ADD15 , _TSFREGIONOFFSET(unsigned int, end ) }, { GEN_INT | GEN_INT_LIMIT960 , _TSFREGIONOFFSET( int, modLfoToVolume ) }, { 0 , (0 ) }, { 0 , (0 ) }, { 0 , (0 ) }, { GEN_FLOAT | GEN_FLOAT_LIMITPAN , _TSFREGIONOFFSET( float, pan ) }, { 0 , (0 ) }, { 0 , (0 ) }, { 0 , (0 ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K5K , _TSFREGIONOFFSET( float, delayModLFO ) }, { GEN_INT | GEN_INT_LIMIT16K4500 , _TSFREGIONOFFSET( int, freqModLFO ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K5K , _TSFREGIONOFFSET( float, delayVibLFO ) }, { GEN_INT | GEN_INT_LIMIT16K4500 , _TSFREGIONOFFSET( int, freqVibLFO ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K5K , _TSFREGIONENVOFFSET( float, modenv, delay ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K8K , _TSFREGIONENVOFFSET( float, modenv, attack ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K5K , _TSFREGIONENVOFFSET( float, modenv, hold ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K8K , _TSFREGIONENVOFFSET( float, modenv, decay ) }, { GEN_FLOAT | GEN_FLOAT_MAX1000 , _TSFREGIONENVOFFSET( float, modenv, sustain ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K8K , _TSFREGIONENVOFFSET( float, modenv, release ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT1200 , _TSFREGIONENVOFFSET( float, modenv, keynumToHold ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT1200 , _TSFREGIONENVOFFSET( float, modenv, keynumToDecay) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K5K , _TSFREGIONENVOFFSET( float, ampenv, delay ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K8K , _TSFREGIONENVOFFSET( float, ampenv, attack ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K5K , _TSFREGIONENVOFFSET( float, ampenv, hold ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K8K , _TSFREGIONENVOFFSET( float, ampenv, decay ) }, { GEN_FLOAT | GEN_FLOAT_MAX1440 , _TSFREGIONENVOFFSET( float, ampenv, sustain ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT12K8K , _TSFREGIONENVOFFSET( float, ampenv, release ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT1200 , _TSFREGIONENVOFFSET( float, ampenv, keynumToHold ) }, { GEN_FLOAT | GEN_FLOAT_LIMIT1200 , _TSFREGIONENVOFFSET( float, ampenv, keynumToDecay) }, { 0 , (0 ) }, { 0 , (0 ) }, { GEN_KEYRANGE , (0 ) }, { GEN_VELRANGE , (0 ) }, { GEN_UINT_ADD15 , _TSFREGIONOFFSET(unsigned int, loop_start ) }, { 0 , (0 ) }, { 0 , (0 ) }, { GEN_FLOAT | GEN_FLOAT_LIMITATTN , _TSFREGIONOFFSET( float, attenuation ) }, { 0 , (0 ) }, { GEN_UINT_ADD15 , _TSFREGIONOFFSET(unsigned int, loop_end ) }, { GEN_INT , _TSFREGIONOFFSET( int, transpose ) }, { GEN_INT , _TSFREGIONOFFSET( int, tune ) }, { 0 , (0 ) }, { GEN_LOOPMODE , _TSFREGIONOFFSET( int, loop_mode ) }, { 0 , (0 ) }, { GEN_INT , _TSFREGIONOFFSET( int, pitch_keytrack ) }, { GEN_GROUP , _TSFREGIONOFFSET(unsigned int, group ) }, { GEN_KEYCENTER , _TSFREGIONOFFSET( int, pitch_keycenter ) }, };
#undef _TSFREGIONOFFSET
#undef _TSFREGIONENVOFFSET
if (amount)
{
int offset;
if (genOper >= _GEN_MAX) return;
offset = genMetas[genOper].offset;
switch (genMetas[genOper].mode & _GEN_TYPE_MASK)
{
case GEN_FLOAT: (( float*)region)[offset] = amount->shortAmount; return;
case GEN_INT: (( int*)region)[offset] = amount->shortAmount; return;
case GEN_UINT_ADD: ((unsigned int*)region)[offset] += amount->shortAmount; return;
case GEN_UINT_ADD15: ((unsigned int*)region)[offset] += amount->shortAmount<<15; return;
case GEN_KEYRANGE: region->lokey = amount->range.lo; region->hikey = amount->range.hi; return;
case GEN_VELRANGE: region->lovel = amount->range.lo; region->hivel = amount->range.hi; return;
case GEN_LOOPMODE: region->loop_mode = ((amount->wordAmount&3) == 3 ? TSF_LOOPMODE_SUSTAIN : ((amount->wordAmount&3) == 1 ? TSF_LOOPMODE_CONTINUOUS : TSF_LOOPMODE_NONE)); return;
case GEN_GROUP: region->group = amount->wordAmount; return;
case GEN_KEYCENTER: region->pitch_keycenter = amount->shortAmount; return;
}
}
else {
for (genOper = 0; genOper != _GEN_MAX; genOper++)
{
int offset = genMetas[genOper].offset;
switch (genMetas[genOper].mode & _GEN_TYPE_MASK)
{
case GEN_FLOAT:
{
float *val = &((float*)region)[offset], vfactor, vmin, vmax;
*val += ((float*)merge_region)[offset];
switch (genMetas[genOper].mode & _GEN_LIMIT_MASK)
{
case GEN_FLOAT_LIMIT12K5K: vfactor = 1.0f; vmin = -12000.0f; vmax = 5000.0f; break;
case GEN_FLOAT_LIMIT12K8K: vfactor = 1.0f; vmin = -12000.0f; vmax = 8000.0f; break;
case GEN_FLOAT_LIMIT1200: vfactor = 1.0f; vmin = -1200.0f; vmax = 1200.0f; break;
case GEN_FLOAT_LIMITPAN: vfactor = 0.001f; vmin = -0.5f; vmax = 0.5f; break;
case GEN_FLOAT_LIMITATTN: vfactor = 0.1f; vmin = 0.0f; vmax = 144.0f; break;
case GEN_FLOAT_MAX1000: vfactor = 1.0f; vmin = 0.0f; vmax = 1000.0f; break;
case GEN_FLOAT_MAX1440: vfactor = 1.0f; vmin = 0.0f; vmax = 1440.0f; break;
default: continue;
}
*val *= vfactor;
if (*val < vmin) *val = vmin;
else if (*val > vmax) *val = vmax;
continue;
}
case GEN_INT:
{
int *val = &((int*)region)[offset], vmin, vmax;
*val += ((int*)merge_region)[offset];
switch (genMetas[genOper].mode & _GEN_LIMIT_MASK)
{
case GEN_INT_LIMIT12K: vmin = -12000; vmax = 12000; break;
case GEN_INT_LIMITFC: vmin = 1500; vmax = 13500; break;
case GEN_INT_LIMITQ: vmin = 0; vmax = 960; break;
case GEN_INT_LIMIT960: vmin = -960; vmax = 960; break;
case GEN_INT_LIMIT16K4500: vmin = -16000; vmax = 4500; break;
default: continue;
}
if (*val < vmin) *val = vmin;
else if (*val > vmax) *val = vmax;
continue;
}
case GEN_UINT_ADD:
{
((unsigned int*)region)[offset] += ((unsigned int*)merge_region)[offset];
continue;
}
}
}
}
}
static void tsf_region_envtosecs(struct tsf_envelope* p, TSF_BOOL sustainIsGain)
{
p->delay = (p->delay < -11950.0f ? 0.0f : tsf_timecents2Secsf(p->delay));
p->attack = (p->attack < -11950.0f ? 0.0f : tsf_timecents2Secsf(p->attack));
p->release = (p->release < -11950.0f ? 0.0f : tsf_timecents2Secsf(p->release));
if (!p->keynumToHold) p->hold = (p->hold < -11950.0f ? 0.0f : tsf_timecents2Secsf(p->hold));
if (!p->keynumToDecay) p->decay = (p->decay < -11950.0f ? 0.0f : tsf_timecents2Secsf(p->decay));
if (p->sustain < 0.0f) p->sustain = 0.0f;
else if (sustainIsGain) p->sustain = tsf_decibelsToGain(-p->sustain / 10.0f);
else p->sustain = 1.0f - (p->sustain / 1000.0f);
}
static void tsf_load_presets(tsf* res, struct tsf_hydra *hydra, unsigned int fontSampleCount)
{
enum { GenInstrument = 41, GenKeyRange = 43, GenVelRange = 44, GenSampleID = 53 };
struct tsf_hydra_phdr *pphdr, *pphdrMax;
for (pphdr = hydra->phdrs, pphdrMax = pphdr + hydra->phdrNum - 1; pphdr != pphdrMax; pphdr++)
{
int sortedIndex = 0, region_index = 0;
struct tsf_hydra_phdr *otherphdr;
struct tsf_preset* preset;
struct tsf_hydra_pbag *ppbag, *ppbagEnd;
struct tsf_region globalRegion;
for (otherphdr = hydra->phdrs; otherphdr != pphdrMax; otherphdr++)
{
if (otherphdr == pphdr || otherphdr->bank > pphdr->bank) continue;
else if (otherphdr->bank < pphdr->bank) sortedIndex++;
else if (otherphdr->preset > pphdr->preset) continue;
else if (otherphdr->preset < pphdr->preset) sortedIndex++;
else if (otherphdr < pphdr) sortedIndex++;
}
preset = &res->presets[sortedIndex];
TSF_MEMCPY(preset->presetName, pphdr->presetName, sizeof(preset->presetName));
preset->presetName[sizeof(preset->presetName)-1] = '\0'; preset->bank = pphdr->bank;
preset->preset = pphdr->preset;
preset->regionNum = 0;
for (ppbag = hydra->pbags + pphdr->presetBagNdx, ppbagEnd = hydra->pbags + pphdr[1].presetBagNdx; ppbag != ppbagEnd; ppbag++)
{
unsigned char plokey = 0, phikey = 127, plovel = 0, phivel = 127;
struct tsf_hydra_pgen *ppgen, *ppgenEnd; struct tsf_hydra_inst *pinst; struct tsf_hydra_ibag *pibag, *pibagEnd; struct tsf_hydra_igen *pigen, *pigenEnd;
for (ppgen = hydra->pgens + ppbag->genNdx, ppgenEnd = hydra->pgens + ppbag[1].genNdx; ppgen != ppgenEnd; ppgen++)
{
if (ppgen->genOper == GenKeyRange) { plokey = ppgen->genAmount.range.lo; phikey = ppgen->genAmount.range.hi; continue; }
if (ppgen->genOper == GenVelRange) { plovel = ppgen->genAmount.range.lo; phivel = ppgen->genAmount.range.hi; continue; }
if (ppgen->genOper != GenInstrument) continue;
if (ppgen->genAmount.wordAmount >= hydra->instNum) continue;
pinst = hydra->insts + ppgen->genAmount.wordAmount;
for (pibag = hydra->ibags + pinst->instBagNdx, pibagEnd = hydra->ibags + pinst[1].instBagNdx; pibag != pibagEnd; pibag++)
{
unsigned char ilokey = 0, ihikey = 127, ilovel = 0, ihivel = 127;
for (pigen = hydra->igens + pibag->instGenNdx, pigenEnd = hydra->igens + pibag[1].instGenNdx; pigen != pigenEnd; pigen++)
{
if (pigen->genOper == GenKeyRange) { ilokey = pigen->genAmount.range.lo; ihikey = pigen->genAmount.range.hi; continue; }
if (pigen->genOper == GenVelRange) { ilovel = pigen->genAmount.range.lo; ihivel = pigen->genAmount.range.hi; continue; }
if (pigen->genOper == GenSampleID && ihikey >= plokey && ilokey <= phikey && ihivel >= plovel && ilovel <= phivel) preset->regionNum++;
}
}
}
}
preset->regions = (struct tsf_region*)TSF_MALLOC(preset->regionNum * sizeof(struct tsf_region));
tsf_region_clear(&globalRegion, TSF_TRUE);
for (ppbag = hydra->pbags + pphdr->presetBagNdx, ppbagEnd = hydra->pbags + pphdr[1].presetBagNdx; ppbag != ppbagEnd; ppbag++)
{
struct tsf_hydra_pgen *ppgen, *ppgenEnd; struct tsf_hydra_inst *pinst; struct tsf_hydra_ibag *pibag, *pibagEnd; struct tsf_hydra_igen *pigen, *pigenEnd;
struct tsf_region presetRegion = globalRegion;
int hadGenInstrument = 0;
for (ppgen = hydra->pgens + ppbag->genNdx, ppgenEnd = hydra->pgens + ppbag[1].genNdx; ppgen != ppgenEnd; ppgen++)
{
if (ppgen->genOper == GenInstrument)
{
struct tsf_region instRegion;
tsf_u16 whichInst = ppgen->genAmount.wordAmount;
if (whichInst >= hydra->instNum) continue;
tsf_region_clear(&instRegion, TSF_FALSE);
pinst = &hydra->insts[whichInst];
for (pibag = hydra->ibags + pinst->instBagNdx, pibagEnd = hydra->ibags + pinst[1].instBagNdx; pibag != pibagEnd; pibag++)
{
struct tsf_region zoneRegion = instRegion;
int hadSampleID = 0;
for (pigen = hydra->igens + pibag->instGenNdx, pigenEnd = hydra->igens + pibag[1].instGenNdx; pigen != pigenEnd; pigen++)
{
if (pigen->genOper == GenSampleID)
{
struct tsf_hydra_shdr* pshdr;
if (zoneRegion.hikey < presetRegion.lokey || zoneRegion.lokey > presetRegion.hikey) continue;
if (zoneRegion.hivel < presetRegion.lovel || zoneRegion.lovel > presetRegion.hivel) continue;
if (presetRegion.lokey > zoneRegion.lokey) zoneRegion.lokey = presetRegion.lokey;
if (presetRegion.hikey < zoneRegion.hikey) zoneRegion.hikey = presetRegion.hikey;
if (presetRegion.lovel > zoneRegion.lovel) zoneRegion.lovel = presetRegion.lovel;
if (presetRegion.hivel < zoneRegion.hivel) zoneRegion.hivel = presetRegion.hivel;
tsf_region_operator(&zoneRegion, 0, TSF_NULL, &presetRegion);
tsf_region_envtosecs(&zoneRegion.ampenv, TSF_TRUE);
tsf_region_envtosecs(&zoneRegion.modenv, TSF_FALSE);
zoneRegion.delayModLFO = (zoneRegion.delayModLFO < -11950.0f ? 0.0f : tsf_timecents2Secsf(zoneRegion.delayModLFO));
zoneRegion.delayVibLFO = (zoneRegion.delayVibLFO < -11950.0f ? 0.0f : tsf_timecents2Secsf(zoneRegion.delayVibLFO));
pshdr = &hydra->shdrs[pigen->genAmount.wordAmount];
zoneRegion.offset += pshdr->start;
zoneRegion.end += pshdr->end;
zoneRegion.loop_start += pshdr->startLoop;
zoneRegion.loop_end += pshdr->endLoop;
if (pshdr->endLoop > 0) zoneRegion.loop_end -= 1;
if (zoneRegion.pitch_keycenter == -1) zoneRegion.pitch_keycenter = pshdr->originalPitch;
zoneRegion.tune += pshdr->pitchCorrection;
zoneRegion.sample_rate = pshdr->sampleRate;
if (zoneRegion.end && zoneRegion.end < fontSampleCount) zoneRegion.end++;
else zoneRegion.end = fontSampleCount;
preset->regions[region_index] = zoneRegion;
region_index++;
hadSampleID = 1;
}
else tsf_region_operator(&zoneRegion, pigen->genOper, &pigen->genAmount, TSF_NULL);
}
if (pibag == hydra->ibags + pinst->instBagNdx && !hadSampleID)
instRegion = zoneRegion;
}
hadGenInstrument = 1;
}
else tsf_region_operator(&presetRegion, ppgen->genOper, &ppgen->genAmount, TSF_NULL);
}
if (ppbag == hydra->pbags + pphdr->presetBagNdx && !hadGenInstrument)
globalRegion = presetRegion;
}
}
}
static void tsf_load_samples(float** fontSamples, unsigned int* fontSampleCount, struct tsf_riffchunk *chunkSmpl, struct tsf_stream* stream)
{
float* out; unsigned int samplesLeft, samplesToRead, samplesToConvert;
samplesLeft = *fontSampleCount = chunkSmpl->size / sizeof(short);
out = *fontSamples = (float*)TSF_MALLOC(samplesLeft * sizeof(float));
for (; samplesLeft; samplesLeft -= samplesToRead)
{
short sampleBuffer[1024], *in = sampleBuffer;;
samplesToRead = (samplesLeft > 1024 ? 1024 : samplesLeft);
stream->read(stream->data, sampleBuffer, samplesToRead * sizeof(short));
for (samplesToConvert = samplesToRead; samplesToConvert > 0; --samplesToConvert)
*out++ = (float)(*in++ / 32767.0);
}
}
static void tsf_voice_envelope_nextsegment(struct tsf_voice_envelope* e, short active_segment, float outSampleRate)
{
switch (active_segment)
{
case TSF_SEGMENT_NONE:
e->samplesUntilNextSegment = (int)(e->parameters.delay * outSampleRate);
if (e->samplesUntilNextSegment > 0)
{
e->segment = TSF_SEGMENT_DELAY;
e->segmentIsExponential = TSF_FALSE;
e->level = 0.0;
e->slope = 0.0;
return;
}
case TSF_SEGMENT_DELAY:
e->samplesUntilNextSegment = (int)(e->parameters.attack * outSampleRate);
if (e->samplesUntilNextSegment > 0)
{
if (!e->isAmpEnv)
{
e->samplesUntilNextSegment = (int)(e->parameters.attack * ((145 - e->midiVelocity) / 144.0f) * outSampleRate);
}
e->segment = TSF_SEGMENT_ATTACK;
e->segmentIsExponential = TSF_FALSE;
e->level = 0.0f;
e->slope = 1.0f / e->samplesUntilNextSegment;
return;
}
case TSF_SEGMENT_ATTACK:
e->samplesUntilNextSegment = (int)(e->parameters.hold * outSampleRate);
if (e->samplesUntilNextSegment > 0)
{
e->segment = TSF_SEGMENT_HOLD;
e->segmentIsExponential = TSF_FALSE;
e->level = 1.0f;
e->slope = 0.0f;
return;
}
case TSF_SEGMENT_HOLD:
e->samplesUntilNextSegment = (int)(e->parameters.decay * outSampleRate);
if (e->samplesUntilNextSegment > 0)
{
e->segment = TSF_SEGMENT_DECAY;
e->level = 1.0f;
if (e->isAmpEnv)
{
float mysterySlope = -9.226f / e->samplesUntilNextSegment;
e->slope = TSF_EXPF(mysterySlope);
e->segmentIsExponential = TSF_TRUE;
if (e->parameters.sustain > 0.0f)
{
e->samplesUntilNextSegment = (int)(TSF_LOG(e->parameters.sustain) / mysterySlope);
}
}
else
{
e->slope = -1.0f / e->samplesUntilNextSegment;
e->samplesUntilNextSegment = (int)(e->parameters.decay * (1.0f - e->parameters.sustain) * outSampleRate);
e->segmentIsExponential = TSF_FALSE;
}
return;
}
case TSF_SEGMENT_DECAY:
e->segment = TSF_SEGMENT_SUSTAIN;
e->level = e->parameters.sustain;
e->slope = 0.0f;
e->samplesUntilNextSegment = 0x7FFFFFFF;
e->segmentIsExponential = TSF_FALSE;
return;
case TSF_SEGMENT_SUSTAIN:
e->segment = TSF_SEGMENT_RELEASE;
e->samplesUntilNextSegment = (int)((e->parameters.release <= 0 ? TSF_FASTRELEASETIME : e->parameters.release) * outSampleRate);
if (e->isAmpEnv)
{
float mysterySlope = -9.226f / e->samplesUntilNextSegment;
e->slope = TSF_EXPF(mysterySlope);
e->segmentIsExponential = TSF_TRUE;
}
else
{
e->slope = -e->level / e->samplesUntilNextSegment;
e->segmentIsExponential = TSF_FALSE;
}
return;
case TSF_SEGMENT_RELEASE:
default:
e->segment = TSF_SEGMENT_DONE;
e->segmentIsExponential = TSF_FALSE;
e->level = e->slope = 0.0f;
e->samplesUntilNextSegment = 0x7FFFFFF;
}
}
static void tsf_voice_envelope_setup(struct tsf_voice_envelope* e, struct tsf_envelope* new_parameters, int midiNoteNumber, short midiVelocity, TSF_BOOL isAmpEnv, float outSampleRate)
{
e->parameters = *new_parameters;
if (e->parameters.keynumToHold)
{
e->parameters.hold += e->parameters.keynumToHold * (60.0f - midiNoteNumber);
e->parameters.hold = (e->parameters.hold < -10000.0f ? 0.0f : tsf_timecents2Secsf(e->parameters.hold));
}
if (e->parameters.keynumToDecay)
{
e->parameters.decay += e->parameters.keynumToDecay * (60.0f - midiNoteNumber);
e->parameters.decay = (e->parameters.decay < -10000.0f ? 0.0f : tsf_timecents2Secsf(e->parameters.decay));
}
e->midiVelocity = midiVelocity;
e->isAmpEnv = isAmpEnv;
tsf_voice_envelope_nextsegment(e, TSF_SEGMENT_NONE, outSampleRate);
}
static void tsf_voice_envelope_process(struct tsf_voice_envelope* e, int numSamples, float outSampleRate)
{
if (e->slope)
{
if (e->segmentIsExponential) e->level *= TSF_POWF(e->slope, (float)numSamples);
else e->level += (e->slope * numSamples);
}
if ((e->samplesUntilNextSegment -= numSamples) <= 0)
tsf_voice_envelope_nextsegment(e, e->segment, outSampleRate);
}
static void tsf_voice_lowpass_setup(struct tsf_voice_lowpass* e, float Fc)
{
double K = TSF_TAN(TSF_PI * Fc), KK = K * K;
double norm = 1 / (1 + K * e->QInv + KK);
e->a0 = KK * norm;
e->a1 = 2 * e->a0;
e->b1 = 2 * (KK - 1) * norm;
e->b2 = (1 - K * e->QInv + KK) * norm;
}
static float tsf_voice_lowpass_process(struct tsf_voice_lowpass* e, double In)
{
double Out = In * e->a0 + e->z1; e->z1 = In * e->a1 + e->z2 - e->b1 * Out; e->z2 = In * e->a0 - e->b2 * Out; return (float)Out;
}
static void tsf_voice_lfo_setup(struct tsf_voice_lfo* e, float delay, int freqCents, float outSampleRate)
{
e->samplesUntil = (int)(delay * outSampleRate);
e->delta = (4.0f * tsf_cents2Hertz((float)freqCents) / outSampleRate);
e->level = 0;
}
static void tsf_voice_lfo_process(struct tsf_voice_lfo* e, int blockSamples)
{
if (e->samplesUntil > blockSamples) { e->samplesUntil -= blockSamples; return; }
e->level += e->delta * blockSamples;
if (e->level > 1.0f) { e->delta = -e->delta; e->level = 2.0f - e->level; }
else if (e->level < -1.0f) { e->delta = -e->delta; e->level = -2.0f - e->level; }
}
static void tsf_voice_kill(struct tsf_voice* v)
{
v->playingPreset = -1;
}
static void tsf_voice_end(tsf* f, struct tsf_voice* v)
{
int repeats = (f->maxVoiceNum ? 2 : 1);
while (repeats--)
{
tsf_voice_envelope_nextsegment(&v->ampenv, TSF_SEGMENT_SUSTAIN, f->outSampleRate);
tsf_voice_envelope_nextsegment(&v->modenv, TSF_SEGMENT_SUSTAIN, f->outSampleRate);
if (v->region->loop_mode == TSF_LOOPMODE_SUSTAIN)
{
v->loopEnd = v->loopStart;
}
}
}
static void tsf_voice_endquick(tsf* f, struct tsf_voice* v)
{
int repeats = (f->maxVoiceNum ? 2 : 1);
while (repeats--)
{
v->ampenv.parameters.release = 0.0f; tsf_voice_envelope_nextsegment(&v->ampenv, TSF_SEGMENT_SUSTAIN, f->outSampleRate);
v->modenv.parameters.release = 0.0f; tsf_voice_envelope_nextsegment(&v->modenv, TSF_SEGMENT_SUSTAIN, f->outSampleRate);
}
}
static void tsf_voice_calcpitchratio(struct tsf_voice* v, float pitchShift, float outSampleRate)
{
double note = v->playingKey + v->region->transpose + v->region->tune / 100.0;
double adjustedPitch = v->region->pitch_keycenter + (note - v->region->pitch_keycenter) * (v->region->pitch_keytrack / 100.0);
if (pitchShift) adjustedPitch += pitchShift;
v->pitchInputTimecents = adjustedPitch * 100.0;
v->pitchOutputFactor = v->region->sample_rate / (tsf_timecents2Secsd(v->region->pitch_keycenter * 100.0) * outSampleRate);
}
static void tsf_voice_render(tsf* f, struct tsf_voice* v, float* outputBuffer, int numSamples)
{
struct tsf_region* region = v->region;
float* input = f->fontSamples;
float* outL = outputBuffer;
float* outR = (f->outputmode == TSF_STEREO_UNWEAVED ? outL + numSamples : TSF_NULL);
TSF_BOOL updateModEnv = (region->modEnvToPitch || region->modEnvToFilterFc);
TSF_BOOL updateModLFO = (v->modlfo.delta && (region->modLfoToPitch || region->modLfoToFilterFc || region->modLfoToVolume));
TSF_BOOL updateVibLFO = (v->viblfo.delta && (region->vibLfoToPitch));
TSF_BOOL isLooping = (v->loopStart < v->loopEnd);
unsigned int tmpLoopStart = v->loopStart, tmpLoopEnd = v->loopEnd;
double tmpSampleEndDbl = (double)region->end, tmpLoopEndDbl = (double)tmpLoopEnd + 1.0;
double tmpSourceSamplePosition = v->sourceSamplePosition;
struct tsf_voice_lowpass tmpLowpass = v->lowpass;
TSF_BOOL dynamicLowpass = (region->modLfoToFilterFc || region->modEnvToFilterFc);
float tmpSampleRate = f->outSampleRate, tmpInitialFilterFc, tmpModLfoToFilterFc, tmpModEnvToFilterFc;
TSF_BOOL dynamicPitchRatio = (region->modLfoToPitch || region->modEnvToPitch || region->vibLfoToPitch);
double pitchRatio;
float tmpModLfoToPitch, tmpVibLfoToPitch, tmpModEnvToPitch;
TSF_BOOL dynamicGain = (region->modLfoToVolume != 0);
float noteGain = 0, tmpModLfoToVolume;
if (dynamicLowpass) tmpInitialFilterFc = (float)region->initialFilterFc, tmpModLfoToFilterFc = (float)region->modLfoToFilterFc, tmpModEnvToFilterFc = (float)region->modEnvToFilterFc;
else tmpInitialFilterFc = 0, tmpModLfoToFilterFc = 0, tmpModEnvToFilterFc = 0;
if (dynamicPitchRatio) pitchRatio = 0, tmpModLfoToPitch = (float)region->modLfoToPitch, tmpVibLfoToPitch = (float)region->vibLfoToPitch, tmpModEnvToPitch = (float)region->modEnvToPitch;
else pitchRatio = tsf_timecents2Secsd(v->pitchInputTimecents) * v->pitchOutputFactor, tmpModLfoToPitch = 0, tmpVibLfoToPitch = 0, tmpModEnvToPitch = 0;
if (dynamicGain) tmpModLfoToVolume = (float)region->modLfoToVolume * 0.1f;
else noteGain = tsf_decibelsToGain(v->noteGainDB), tmpModLfoToVolume = 0;
while (numSamples)
{
float gainMono, gainLeft, gainRight;
int blockSamples = (numSamples > TSF_RENDER_EFFECTSAMPLEBLOCK ? TSF_RENDER_EFFECTSAMPLEBLOCK : numSamples);
numSamples -= blockSamples;
if (dynamicLowpass)
{
float fres = tmpInitialFilterFc + v->modlfo.level * tmpModLfoToFilterFc + v->modenv.level * tmpModEnvToFilterFc;
float lowpassFc = (fres <= 13500 ? tsf_cents2Hertz(fres) / tmpSampleRate : 1.0f);
tmpLowpass.active = (lowpassFc < 0.499f);
if (tmpLowpass.active) tsf_voice_lowpass_setup(&tmpLowpass, lowpassFc);
}
if (dynamicPitchRatio)
pitchRatio = tsf_timecents2Secsd(v->pitchInputTimecents + (v->modlfo.level * tmpModLfoToPitch + v->viblfo.level * tmpVibLfoToPitch + v->modenv.level * tmpModEnvToPitch)) * v->pitchOutputFactor;
if (dynamicGain)
noteGain = tsf_decibelsToGain(v->noteGainDB + (v->modlfo.level * tmpModLfoToVolume));
gainMono = noteGain * v->ampenv.level;
tsf_voice_envelope_process(&v->ampenv, blockSamples, tmpSampleRate);
if (updateModEnv) tsf_voice_envelope_process(&v->modenv, blockSamples, tmpSampleRate);
if (updateModLFO) tsf_voice_lfo_process(&v->modlfo, blockSamples);
if (updateVibLFO) tsf_voice_lfo_process(&v->viblfo, blockSamples);
switch (f->outputmode)
{
case TSF_STEREO_INTERLEAVED:
gainLeft = gainMono * v->panFactorLeft, gainRight = gainMono * v->panFactorRight;
while (blockSamples-- && tmpSourceSamplePosition < tmpSampleEndDbl)
{
unsigned int pos = (unsigned int)tmpSourceSamplePosition, nextPos = (pos >= tmpLoopEnd && isLooping ? tmpLoopStart : pos + 1);
float alpha = (float)(tmpSourceSamplePosition - pos), val = (input[pos] * (1.0f - alpha) + input[nextPos] * alpha);
if (tmpLowpass.active) val = tsf_voice_lowpass_process(&tmpLowpass, val);
*outL++ += val * gainLeft;
*outL++ += val * gainRight;
tmpSourceSamplePosition += pitchRatio;
if (tmpSourceSamplePosition >= tmpLoopEndDbl && isLooping) tmpSourceSamplePosition -= (tmpLoopEnd - tmpLoopStart + 1.0);
}
break;
case TSF_STEREO_UNWEAVED:
gainLeft = gainMono * v->panFactorLeft, gainRight = gainMono * v->panFactorRight;
while (blockSamples-- && tmpSourceSamplePosition < tmpSampleEndDbl)
{
unsigned int pos = (unsigned int)tmpSourceSamplePosition, nextPos = (pos >= tmpLoopEnd && isLooping ? tmpLoopStart : pos + 1);
float alpha = (float)(tmpSourceSamplePosition - pos), val = (input[pos] * (1.0f - alpha) + input[nextPos] * alpha);
if (tmpLowpass.active) val = tsf_voice_lowpass_process(&tmpLowpass, val);
*outL++ += val * gainLeft;
*outR++ += val * gainRight;
tmpSourceSamplePosition += pitchRatio;
if (tmpSourceSamplePosition >= tmpLoopEndDbl && isLooping) tmpSourceSamplePosition -= (tmpLoopEnd - tmpLoopStart + 1.0);
}
break;
case TSF_MONO:
while (blockSamples-- && tmpSourceSamplePosition < tmpSampleEndDbl)
{
unsigned int pos = (unsigned int)tmpSourceSamplePosition, nextPos = (pos >= tmpLoopEnd && isLooping ? tmpLoopStart : pos + 1);
float alpha = (float)(tmpSourceSamplePosition - pos), val = (input[pos] * (1.0f - alpha) + input[nextPos] * alpha);
if (tmpLowpass.active) val = tsf_voice_lowpass_process(&tmpLowpass, val);
*outL++ += val * gainMono;
tmpSourceSamplePosition += pitchRatio;
if (tmpSourceSamplePosition >= tmpLoopEndDbl && isLooping) tmpSourceSamplePosition -= (tmpLoopEnd - tmpLoopStart + 1.0);
}
break;
}
if (tmpSourceSamplePosition >= tmpSampleEndDbl || v->ampenv.segment == TSF_SEGMENT_DONE)
{
tsf_voice_kill(v);
return;
}
}
v->sourceSamplePosition = tmpSourceSamplePosition;
if (tmpLowpass.active || dynamicLowpass) v->lowpass = tmpLowpass;
}
TSFDEF tsf* tsf_load(struct tsf_stream* stream)
{
tsf* res = TSF_NULL;
struct tsf_riffchunk chunkHead;
struct tsf_riffchunk chunkList;
struct tsf_hydra hydra;
float* fontSamples = TSF_NULL;
unsigned int fontSampleCount = 0;
if (!tsf_riffchunk_read(TSF_NULL, &chunkHead, stream) || !TSF_FourCCEquals(chunkHead.id, "sfbk"))
{
return res;
}
TSF_MEMSET(&hydra, 0, sizeof(hydra));
while (tsf_riffchunk_read(&chunkHead, &chunkList, stream))
{
struct tsf_riffchunk chunk;
if (TSF_FourCCEquals(chunkList.id, "pdta"))
{
while (tsf_riffchunk_read(&chunkList, &chunk, stream))
{
#define HandleChunk(chunkName) (TSF_FourCCEquals(chunk.id, #chunkName) && !(chunk.size % chunkName##SizeInFile)) \
{ \
int num = chunk.size / chunkName##SizeInFile, i; \
hydra.chunkName##Num = num; \
hydra.chunkName##s = (struct tsf_hydra_##chunkName*)TSF_MALLOC(num * sizeof(struct tsf_hydra_##chunkName)); \
for (i = 0; i < num; ++i) tsf_hydra_read_##chunkName(&hydra.chunkName##s[i], stream); \
}
enum
{
phdrSizeInFile = 38, pbagSizeInFile = 4, pmodSizeInFile = 10,
pgenSizeInFile = 4, instSizeInFile = 22, ibagSizeInFile = 4,
imodSizeInFile = 10, igenSizeInFile = 4, shdrSizeInFile = 46
};
if HandleChunk(phdr) else if HandleChunk(pbag) else if HandleChunk(pmod)
else if HandleChunk(pgen) else if HandleChunk(inst) else if HandleChunk(ibag)
else if HandleChunk(imod) else if HandleChunk(igen) else if HandleChunk(shdr)
else stream->skip(stream->data, chunk.size);
#undef HandleChunk
}
}
else if (TSF_FourCCEquals(chunkList.id, "sdta"))
{
while (tsf_riffchunk_read(&chunkList, &chunk, stream))
{
if (TSF_FourCCEquals(chunk.id, "smpl"))
{
tsf_load_samples(&fontSamples, &fontSampleCount, &chunk, stream);
}
else stream->skip(stream->data, chunk.size);
}
}
else stream->skip(stream->data, chunkList.size);
}
if (!hydra.phdrs || !hydra.pbags || !hydra.pmods || !hydra.pgens || !hydra.insts || !hydra.ibags || !hydra.imods || !hydra.igens || !hydra.shdrs)
{
}
else if (fontSamples == TSF_NULL)
{
}
else
{
res = (tsf*)TSF_MALLOC(sizeof(tsf));
TSF_MEMSET(res, 0, sizeof(tsf));
res->presetNum = hydra.phdrNum - 1;
res->presets = (struct tsf_preset*)TSF_MALLOC(res->presetNum * sizeof(struct tsf_preset));
res->fontSamples = fontSamples;
res->outSampleRate = 44100.0f;
fontSamples = TSF_NULL; tsf_load_presets(res, &hydra, fontSampleCount);
}
TSF_FREE(hydra.phdrs); TSF_FREE(hydra.pbags); TSF_FREE(hydra.pmods);
TSF_FREE(hydra.pgens); TSF_FREE(hydra.insts); TSF_FREE(hydra.ibags);
TSF_FREE(hydra.imods); TSF_FREE(hydra.igens); TSF_FREE(hydra.shdrs);
TSF_FREE(fontSamples);
return res;
}
TSFDEF void tsf_close(tsf* f)
{
struct tsf_preset *preset, *presetEnd;
if (!f) return;
for (preset = f->presets, presetEnd = preset + f->presetNum; preset != presetEnd; preset++)
TSF_FREE(preset->regions);
TSF_FREE(f->presets);
TSF_FREE(f->fontSamples);
TSF_FREE(f->voices);
if (f->channels) { TSF_FREE(f->channels->channels); TSF_FREE(f->channels); }
TSF_FREE(f->outputSamples);
TSF_FREE(f);
}
TSFDEF void tsf_reset(tsf* f)
{
struct tsf_voice *v = f->voices, *vEnd = v + f->voiceNum;
for (; v != vEnd; v++)
if (v->playingPreset != -1 && (v->ampenv.segment < TSF_SEGMENT_RELEASE || v->ampenv.parameters.release))
tsf_voice_endquick(f, v);
if (f->channels) { TSF_FREE(f->channels->channels); TSF_FREE(f->channels); f->channels = TSF_NULL; }
}
TSFDEF int tsf_get_presetindex(const tsf* f, int bank, int preset_number)
{
const struct tsf_preset *presets;
int i, iMax;
for (presets = f->presets, i = 0, iMax = f->presetNum; i < iMax; i++)
if (presets[i].preset == preset_number && presets[i].bank == bank)
return i;
return -1;
}
TSFDEF int tsf_get_presetcount(const tsf* f)
{
return f->presetNum;
}
TSFDEF const char* tsf_get_presetname(const tsf* f, int preset)
{
return (preset < 0 || preset >= f->presetNum ? TSF_NULL : f->presets[preset].presetName);
}
TSFDEF const char* tsf_bank_get_presetname(const tsf* f, int bank, int preset_number)
{
return tsf_get_presetname(f, tsf_get_presetindex(f, bank, preset_number));
}
TSFDEF void tsf_set_output(tsf* f, enum TSFOutputMode outputmode, int samplerate, float global_gain_db)
{
f->outputmode = outputmode;
f->outSampleRate = (float)(samplerate >= 1 ? samplerate : 44100.0f);
f->globalGainDB = global_gain_db;
}
TSFDEF void tsf_set_volume(tsf* f, float global_volume)
{
f->globalGainDB = (global_volume == 1.0f ? 0 : -tsf_gainToDecibels(1.0f / global_volume));
}
TSFDEF void tsf_set_max_voices(tsf* f, int max_voices)
{
int i = f->voiceNum;
f->voiceNum = f->maxVoiceNum = (f->voiceNum > max_voices ? f->voiceNum : max_voices);
f->voices = (struct tsf_voice*)TSF_REALLOC(f->voices, f->voiceNum * sizeof(struct tsf_voice));
for (; i != max_voices; i++)
f->voices[i].playingPreset = -1;
}
TSFDEF void tsf_note_on(tsf* f, int preset_index, int key, float vel)
{
short midiVelocity = (short)(vel * 127);
int voicePlayIndex;
struct tsf_region *region, *regionEnd;
if (preset_index < 0 || preset_index >= f->presetNum) return;
if (vel <= 0.0f) { tsf_note_off(f, preset_index, key); return; }
voicePlayIndex = f->voicePlayIndex++;
for (region = f->presets[preset_index].regions, regionEnd = region + f->presets[preset_index].regionNum; region != regionEnd; region++)
{
struct tsf_voice *voice, *v, *vEnd; TSF_BOOL doLoop; float lowpassFilterQDB, lowpassFc;
if (key < region->lokey || key > region->hikey || midiVelocity < region->lovel || midiVelocity > region->hivel) continue;
voice = TSF_NULL, v = f->voices, vEnd = v + f->voiceNum;
if (region->group)
{
for (; v != vEnd; v++)
if (v->playingPreset == preset_index && v->region->group == region->group) tsf_voice_endquick(f, v);
else if (v->playingPreset == -1 && !voice) voice = v;
}
else for (; v != vEnd; v++) if (v->playingPreset == -1) { voice = v; break; }
if (!voice)
{
if (f->maxVoiceNum)
{
continue;
}
f->voiceNum += 4;
f->voices = (struct tsf_voice*)TSF_REALLOC(f->voices, f->voiceNum * sizeof(struct tsf_voice));
voice = &f->voices[f->voiceNum - 4];
voice[1].playingPreset = voice[2].playingPreset = voice[3].playingPreset = -1;
}
voice->region = region;
voice->playingPreset = preset_index;
voice->playingKey = key;
voice->playIndex = voicePlayIndex;
voice->noteGainDB = f->globalGainDB - region->attenuation - tsf_gainToDecibels(1.0f / vel);
if (f->channels)
{
f->channels->setupVoice(f, voice);
}
else
{
tsf_voice_calcpitchratio(voice, 0, f->outSampleRate);
voice->panFactorLeft = TSF_SQRTF(0.5f - region->pan);
voice->panFactorRight = TSF_SQRTF(0.5f + region->pan);
}
voice->sourceSamplePosition = region->offset;
doLoop = (region->loop_mode != TSF_LOOPMODE_NONE && region->loop_start < region->loop_end);
voice->loopStart = (doLoop ? region->loop_start : 0);
voice->loopEnd = (doLoop ? region->loop_end : 0);
tsf_voice_envelope_setup(&voice->ampenv, ®ion->ampenv, key, midiVelocity, TSF_TRUE, f->outSampleRate);
tsf_voice_envelope_setup(&voice->modenv, ®ion->modenv, key, midiVelocity, TSF_FALSE, f->outSampleRate);
lowpassFc = (region->initialFilterFc <= 13500 ? tsf_cents2Hertz((float)region->initialFilterFc) / f->outSampleRate : 1.0f);
lowpassFilterQDB = region->initialFilterQ / 10.0f;
voice->lowpass.QInv = 1.0 / TSF_POW(10.0, (lowpassFilterQDB / 20.0));
voice->lowpass.z1 = voice->lowpass.z2 = 0;
voice->lowpass.active = (lowpassFc < 0.499f);
if (voice->lowpass.active) tsf_voice_lowpass_setup(&voice->lowpass, lowpassFc);
tsf_voice_lfo_setup(&voice->modlfo, region->delayModLFO, region->freqModLFO, f->outSampleRate);
tsf_voice_lfo_setup(&voice->viblfo, region->delayVibLFO, region->freqVibLFO, f->outSampleRate);
}
}
TSFDEF int tsf_bank_note_on(tsf* f, int bank, int preset_number, int key, float vel)
{
int preset_index = tsf_get_presetindex(f, bank, preset_number);
if (preset_index == -1) return 0;
tsf_note_on(f, preset_index, key, vel);
return 1;
}
TSFDEF void tsf_note_off(tsf* f, int preset_index, int key)
{
struct tsf_voice *v = f->voices, *vEnd = v + f->voiceNum, *vMatchFirst = TSF_NULL, *vMatchLast = TSF_NULL;
for (; v != vEnd; v++)
{
if (v->playingPreset != preset_index || v->playingKey != key || v->ampenv.segment >= TSF_SEGMENT_RELEASE) continue;
else if (!vMatchFirst || v->playIndex < vMatchFirst->playIndex) vMatchFirst = vMatchLast = v;
else if (v->playIndex == vMatchFirst->playIndex) vMatchLast = v;
}
if (!vMatchFirst) return;
for (v = vMatchFirst; v <= vMatchLast; v++)
{
if (v != vMatchFirst && v != vMatchLast &&
(v->playIndex != vMatchFirst->playIndex || v->playingPreset != preset_index || v->playingKey != key || v->ampenv.segment >= TSF_SEGMENT_RELEASE)) continue;
tsf_voice_end(f, v);
}
}
TSFDEF int tsf_bank_note_off(tsf* f, int bank, int preset_number, int key)
{
int preset_index = tsf_get_presetindex(f, bank, preset_number);
if (preset_index == -1) return 0;
tsf_note_off(f, preset_index, key);
return 1;
}
TSFDEF void tsf_note_off_all(tsf* f)
{
struct tsf_voice *v = f->voices, *vEnd = v + f->voiceNum;
for (; v != vEnd; v++) if (v->playingPreset != -1 && v->ampenv.segment < TSF_SEGMENT_RELEASE)
tsf_voice_end(f, v);
}
TSFDEF int tsf_active_voice_count(tsf* f)
{
int count = 0;
struct tsf_voice *v = f->voices, *vEnd = v + f->voiceNum;
for (; v != vEnd; v++) if (v->playingPreset != -1) count++;
return count;
}
TSFDEF void tsf_render_short(tsf* f, short* buffer, int samples, int flag_mixing)
{
float *floatSamples;
int channelSamples = (f->outputmode == TSF_MONO ? 1 : 2) * samples, floatBufferSize = channelSamples * sizeof(float);
short* bufferEnd = buffer + channelSamples;
if (floatBufferSize > f->outputSampleSize)
{
TSF_FREE(f->outputSamples);
f->outputSamples = (float*)TSF_MALLOC(floatBufferSize);
f->outputSampleSize = floatBufferSize;
}
tsf_render_float(f, f->outputSamples, samples, TSF_FALSE);
floatSamples = f->outputSamples;
if (flag_mixing)
while (buffer != bufferEnd)
{
float v = *floatSamples++;
int vi = *buffer + (v < -1.00004566f ? (int)-32768 : (v > 1.00001514f ? (int)32767 : (int)(v * 32767.5f)));
*buffer++ = (vi < -32768 ? (short)-32768 : (vi > 32767 ? (short)32767 : (short)vi));
}
else
while (buffer != bufferEnd)
{
float v = *floatSamples++;
*buffer++ = (v < -1.00004566f ? (short)-32768 : (v > 1.00001514f ? (short)32767 : (short)(v * 32767.5f)));
}
}
TSFDEF void tsf_render_float(tsf* f, float* buffer, int samples, int flag_mixing)
{
struct tsf_voice *v = f->voices, *vEnd = v + f->voiceNum;
if (!flag_mixing) TSF_MEMSET(buffer, 0, (f->outputmode == TSF_MONO ? 1 : 2) * sizeof(float) * samples);
for (; v != vEnd; v++)
if (v->playingPreset != -1)
tsf_voice_render(f, v, buffer, samples);
}
static void tsf_channel_setup_voice(tsf* f, struct tsf_voice* v)
{
struct tsf_channel* c = &f->channels->channels[f->channels->activeChannel];
float newpan = v->region->pan + c->panOffset;
v->playingChannel = f->channels->activeChannel;
v->noteGainDB += c->gainDB;
tsf_voice_calcpitchratio(v, (c->pitchWheel == 8192 ? c->tuning : ((c->pitchWheel / 16383.0f * c->pitchRange * 2.0f) - c->pitchRange + c->tuning)), f->outSampleRate);
if (newpan <= -0.5f) { v->panFactorLeft = 1.0f; v->panFactorRight = 0.0f; }
else if (newpan >= 0.5f) { v->panFactorLeft = 0.0f; v->panFactorRight = 1.0f; }
else { v->panFactorLeft = TSF_SQRTF(0.5f - newpan); v->panFactorRight = TSF_SQRTF(0.5f + newpan); }
}
static struct tsf_channel* tsf_channel_init(tsf* f, int channel)
{
int i;
if (f->channels && channel < f->channels->channelNum) return &f->channels->channels[channel];
if (!f->channels)
{
f->channels = (struct tsf_channels*)TSF_MALLOC(sizeof(struct tsf_channels));
f->channels->setupVoice = &tsf_channel_setup_voice;
f->channels->channels = NULL;
f->channels->channelNum = 0;
f->channels->activeChannel = 0;
}
i = f->channels->channelNum;
f->channels->channelNum = channel + 1;
f->channels->channels = (struct tsf_channel*)TSF_REALLOC(f->channels->channels, f->channels->channelNum * sizeof(struct tsf_channel));
for (; i <= channel; i++)
{
struct tsf_channel* c = &f->channels->channels[i];
c->presetIndex = c->bank = 0;
c->pitchWheel = c->midiPan = 8192;
c->midiVolume = c->midiExpression = 16383;
c->midiRPN = 0xFFFF;
c->midiData = 0;
c->panOffset = 0.0f;
c->gainDB = 0.0f;
c->pitchRange = 2.0f;
c->tuning = 0.0f;
}
return &f->channels->channels[channel];
}
static void tsf_channel_applypitch(tsf* f, int channel, struct tsf_channel* c)
{
struct tsf_voice *v, *vEnd;
float pitchShift = (c->pitchWheel == 8192 ? c->tuning : ((c->pitchWheel / 16383.0f * c->pitchRange * 2.0f) - c->pitchRange + c->tuning));
for (v = f->voices, vEnd = v + f->voiceNum; v != vEnd; v++)
if (v->playingChannel == channel && v->playingPreset != -1)
tsf_voice_calcpitchratio(v, pitchShift, f->outSampleRate);
}
TSFDEF void tsf_channel_set_presetindex(tsf* f, int channel, int preset_index)
{
tsf_channel_init(f, channel)->presetIndex = (unsigned short)preset_index;
}
TSFDEF int tsf_channel_set_presetnumber(tsf* f, int channel, int preset_number, int flag_mididrums)
{
struct tsf_channel *c = tsf_channel_init(f, channel);
int preset_index;
if (flag_mididrums)
{
preset_index = tsf_get_presetindex(f, 128 | (c->bank & 0x7FFF), preset_number);
if (preset_index == -1) preset_index = tsf_get_presetindex(f, 128, preset_number);
if (preset_index == -1) preset_index = tsf_get_presetindex(f, 128, 0);
if (preset_index == -1) preset_index = tsf_get_presetindex(f, (c->bank & 0x7FFF), preset_number);
}
else preset_index = tsf_get_presetindex(f, (c->bank & 0x7FFF), preset_number);
if (preset_index == -1) preset_index = tsf_get_presetindex(f, 0, preset_number);
if (preset_index != -1)
{
c->presetIndex = (unsigned short)preset_index;
return 1;
}
return 0;
}
TSFDEF void tsf_channel_set_bank(tsf* f, int channel, int bank)
{
tsf_channel_init(f, channel)->bank = (unsigned short)bank;
}
TSFDEF int tsf_channel_set_bank_preset(tsf* f, int channel, int bank, int preset_number)
{
struct tsf_channel *c = tsf_channel_init(f, channel);
int preset_index = tsf_get_presetindex(f, bank, preset_number);
if (preset_index == -1) return 0;
c->presetIndex = (unsigned short)preset_index;
c->bank = (unsigned short)bank;
return 1;
}
TSFDEF void tsf_channel_set_pan(tsf* f, int channel, float pan)
{
struct tsf_voice *v, *vEnd;
for (v = f->voices, vEnd = v + f->voiceNum; v != vEnd; v++)
if (v->playingChannel == channel && v->playingPreset != -1)
{
float newpan = v->region->pan + pan - 0.5f;
if (newpan <= -0.5f) { v->panFactorLeft = 1.0f; v->panFactorRight = 0.0f; }
else if (newpan >= 0.5f) { v->panFactorLeft = 0.0f; v->panFactorRight = 1.0f; }
else { v->panFactorLeft = TSF_SQRTF(0.5f - newpan); v->panFactorRight = TSF_SQRTF(0.5f + newpan); }
}
tsf_channel_init(f, channel)->panOffset = pan - 0.5f;
}
TSFDEF void tsf_channel_set_volume(tsf* f, int channel, float volume)
{
struct tsf_channel *c = tsf_channel_init(f, channel);
float gainDB = tsf_gainToDecibels(volume), gainDBChange = gainDB - c->gainDB;
struct tsf_voice *v, *vEnd;
if (gainDBChange == 0) return;
for (v = f->voices, vEnd = v + f->voiceNum; v != vEnd; v++)
if (v->playingChannel == channel && v->playingPreset != -1)
v->noteGainDB += gainDBChange;
c->gainDB = gainDB;
}
TSFDEF void tsf_channel_set_pitchwheel(tsf* f, int channel, int pitch_wheel)
{
struct tsf_channel *c = tsf_channel_init(f, channel);
if (c->pitchWheel == pitch_wheel) return;
c->pitchWheel = (unsigned short)pitch_wheel;
tsf_channel_applypitch(f, channel, c);
}
TSFDEF void tsf_channel_set_pitchrange(tsf* f, int channel, float pitch_range)
{
struct tsf_channel *c = tsf_channel_init(f, channel);
if (c->pitchRange == pitch_range) return;
c->pitchRange = pitch_range;
if (c->pitchWheel != 8192) tsf_channel_applypitch(f, channel, c);
}
TSFDEF void tsf_channel_set_tuning(tsf* f, int channel, float tuning)
{
struct tsf_channel *c = tsf_channel_init(f, channel);
if (c->tuning == tuning) return;
c->tuning = tuning;
tsf_channel_applypitch(f, channel, c);
}
TSFDEF void tsf_channel_note_on(tsf* f, int channel, int key, float vel)
{
if (!f->channels || channel >= f->channels->channelNum) return;
f->channels->activeChannel = channel;
tsf_note_on(f, f->channels->channels[channel].presetIndex, key, vel);
}
TSFDEF void tsf_channel_note_off(tsf* f, int channel, int key)
{
struct tsf_voice *v = f->voices, *vEnd = v + f->voiceNum, *vMatchFirst = TSF_NULL, *vMatchLast = TSF_NULL;
for (; v != vEnd; v++)
{
if (v->playingPreset == -1 || v->playingChannel != channel || v->playingKey != key || v->ampenv.segment >= TSF_SEGMENT_RELEASE) continue;
else if (!vMatchFirst || v->playIndex < vMatchFirst->playIndex) vMatchFirst = vMatchLast = v;
else if (v->playIndex == vMatchFirst->playIndex) vMatchLast = v;
}
if (!vMatchFirst) return;
for (v = vMatchFirst; v <= vMatchLast; v++)
{
if (v != vMatchFirst && v != vMatchLast &&
(v->playIndex != vMatchFirst->playIndex || v->playingPreset == -1 || v->playingChannel != channel || v->playingKey != key || v->ampenv.segment >= TSF_SEGMENT_RELEASE)) continue;
tsf_voice_end(f, v);
}
}
TSFDEF void tsf_channel_note_off_all(tsf* f, int channel)
{
struct tsf_voice *v = f->voices, *vEnd = v + f->voiceNum;
for (; v != vEnd; v++)
if (v->playingPreset != -1 && v->playingChannel == channel && v->ampenv.segment < TSF_SEGMENT_RELEASE)
tsf_voice_end(f, v);
}
TSFDEF void tsf_channel_sounds_off_all(tsf* f, int channel)
{
struct tsf_voice *v = f->voices, *vEnd = v + f->voiceNum;
for (; v != vEnd; v++)
if (v->playingPreset != -1 && v->playingChannel == channel && (v->ampenv.segment < TSF_SEGMENT_RELEASE || v->ampenv.parameters.release))
tsf_voice_endquick(f, v);
}
TSFDEF void tsf_channel_midi_control(tsf* f, int channel, int controller, int control_value)
{
struct tsf_channel* c = tsf_channel_init(f, channel);
switch (controller)
{
case 7 : c->midiVolume = (unsigned short)((c->midiVolume & 0x7F ) | (control_value << 7)); goto TCMC_SET_VOLUME;
case 39 : c->midiVolume = (unsigned short)((c->midiVolume & 0x3F80) | control_value); goto TCMC_SET_VOLUME;
case 11 : c->midiExpression = (unsigned short)((c->midiExpression & 0x7F ) | (control_value << 7)); goto TCMC_SET_VOLUME;
case 43 : c->midiExpression = (unsigned short)((c->midiExpression & 0x3F80) | control_value); goto TCMC_SET_VOLUME;
case 10 : c->midiPan = (unsigned short)((c->midiPan & 0x7F ) | (control_value << 7)); goto TCMC_SET_PAN;
case 42 : c->midiPan = (unsigned short)((c->midiPan & 0x3F80) | control_value); goto TCMC_SET_PAN;
case 6 : c->midiData = (unsigned short)((c->midiData & 0x7F) | (control_value << 7)); goto TCMC_SET_DATA;
case 38 : c->midiData = (unsigned short)((c->midiData & 0x3F80) | control_value); goto TCMC_SET_DATA;
case 0 : c->bank = (unsigned short)(0x8000 | control_value); return; case 32 : c->bank = (unsigned short)((c->bank & 0x8000 ? ((c->bank & 0x7F) << 7) : 0) | control_value); return;
case 101 : c->midiRPN = (unsigned short)(((c->midiRPN == 0xFFFF ? 0 : c->midiRPN) & 0x7F ) | (control_value << 7)); return;
case 100 : c->midiRPN = (unsigned short)(((c->midiRPN == 0xFFFF ? 0 : c->midiRPN) & 0x3F80) | control_value); return;
case 98 : c->midiRPN = 0xFFFF; return;
case 99 : c->midiRPN = 0xFFFF; return;
case 120 : tsf_channel_sounds_off_all(f, channel); return;
case 123 : tsf_channel_note_off_all(f, channel); return;
case 121 :
c->midiVolume = c->midiExpression = 16383;
c->midiPan = 8192;
c->bank = 0;
tsf_channel_set_volume(f, channel, 1.0f);
tsf_channel_set_pan(f, channel, 0.5f);
tsf_channel_set_pitchrange(f, channel, 2.0f);
return;
}
return;
TCMC_SET_VOLUME:
tsf_channel_set_volume(f, channel, TSF_POWF((c->midiVolume / 16383.0f) * (c->midiExpression / 16383.0f), 3.0f));
return;
TCMC_SET_PAN:
tsf_channel_set_pan(f, channel, c->midiPan / 16383.0f);
return;
TCMC_SET_DATA:
if (c->midiRPN == 0) tsf_channel_set_pitchrange(f, channel, (c->midiData >> 7) + 0.01f * (c->midiData & 0x7F));
else if (c->midiRPN == 1) tsf_channel_set_tuning(f, channel, (int)c->tuning + ((float)c->midiData - 8192.0f) / 8192.0f); else if (c->midiRPN == 2 && controller == 6) tsf_channel_set_tuning(f, channel, ((float)control_value - 64.0f) + (c->tuning - (int)c->tuning)); return;
}
TSFDEF int tsf_channel_get_preset_index(tsf* f, int channel)
{
return (f->channels && channel < f->channels->channelNum ? f->channels->channels[channel].presetIndex : 0);
}
TSFDEF int tsf_channel_get_preset_bank(tsf* f, int channel)
{
return (f->channels && channel < f->channels->channelNum ? (f->channels->channels[channel].bank & 0x7FFF) : 0);
}
TSFDEF int tsf_channel_get_preset_number(tsf* f, int channel)
{
return (f->channels && channel < f->channels->channelNum ? f->presets[f->channels->channels[channel].presetIndex].preset : 0);
}
TSFDEF float tsf_channel_get_pan(tsf* f, int channel)
{
return (f->channels && channel < f->channels->channelNum ? f->channels->channels[channel].panOffset - 0.5f : 0.5f);
}
TSFDEF float tsf_channel_get_volume(tsf* f, int channel)
{
return (f->channels && channel < f->channels->channelNum ? tsf_decibelsToGain(f->channels->channels[channel].gainDB) : 1.0f);
}
TSFDEF int tsf_channel_get_pitchwheel(tsf* f, int channel)
{
return (f->channels && channel < f->channels->channelNum ? f->channels->channels[channel].pitchWheel : 8192);
}
TSFDEF float tsf_channel_get_pitchrange(tsf* f, int channel)
{
return (f->channels && channel < f->channels->channelNum ? f->channels->channels[channel].pitchRange : 2.0f);
}
TSFDEF float tsf_channel_get_tuning(tsf* f, int channel)
{
return (f->channels && channel < f->channels->channelNum ? f->channels->channels[channel].tuning : 0.0f);
}
#ifdef __cplusplus
}
#endif
#endif