2010-06-26 15:02:58 +00:00
|
|
|
/* FluidSynth - A Software Synthesizer
|
|
|
|
*
|
|
|
|
* Copyright (C) 2003 Peter Hanappe and others.
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
2017-07-12 15:45:23 +00:00
|
|
|
* modify it under the terms of the GNU Lesser General Public License
|
2017-07-12 15:53:03 +00:00
|
|
|
* as published by the Free Software Foundation; either version 2.1 of
|
2010-06-26 15:02:58 +00:00
|
|
|
* the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed 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
|
2017-07-12 15:45:23 +00:00
|
|
|
* Lesser General Public License for more details.
|
2010-06-26 15:02:58 +00:00
|
|
|
*
|
2017-07-12 15:54:54 +00:00
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
2010-06-26 15:02:58 +00:00
|
|
|
* License along with this library; if not, write to the Free
|
2011-08-15 12:57:10 +00:00
|
|
|
* Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
|
|
|
* 02110-1301, USA
|
2010-06-26 15:02:58 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "fluidsynth_priv.h"
|
|
|
|
#include "fluid_phase.h"
|
|
|
|
#include "fluid_rvoice.h"
|
|
|
|
#include "fluid_sys.h"
|
|
|
|
|
|
|
|
/* Purpose:
|
|
|
|
*
|
|
|
|
* Interpolates audio data (obtains values between the samples of the original
|
|
|
|
* waveform data).
|
|
|
|
*
|
2018-04-08 10:18:38 +00:00
|
|
|
* Variables loaded from the voice structure (assigned in fluid_rvoice_write()):
|
2010-06-26 15:02:58 +00:00
|
|
|
* - dsp_data: Pointer to the original waveform data
|
|
|
|
* - dsp_phase: The position in the original waveform data.
|
|
|
|
* This has an integer and a fractional part (between samples).
|
|
|
|
* - dsp_phase_incr: For each output sample, the position in the original
|
|
|
|
* waveform advances by dsp_phase_incr. This also has an integer
|
|
|
|
* part and a fractional part.
|
|
|
|
* If a sample is played at root pitch (no pitch change),
|
|
|
|
* dsp_phase_incr is integer=1 and fractional=0.
|
|
|
|
* - dsp_amp: The current amplitude envelope value.
|
|
|
|
* - dsp_amp_incr: The changing rate of the amplitude envelope.
|
|
|
|
*
|
|
|
|
* A couple of variables are used internally, their results are discarded:
|
|
|
|
* - dsp_i: Index through the output buffer
|
|
|
|
* - dsp_buf: Output buffer of floating point values (FLUID_BUFSIZE in length)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Interpolation (find a value between two samples of the original waveform) */
|
|
|
|
|
|
|
|
/* Linear interpolation table (2 coefficients centered on 1st) */
|
|
|
|
static fluid_real_t interp_coeff_linear[FLUID_INTERP_MAX][2];
|
|
|
|
|
|
|
|
/* 4th order (cubic) interpolation table (4 coefficients centered on 2nd) */
|
|
|
|
static fluid_real_t interp_coeff[FLUID_INTERP_MAX][4];
|
|
|
|
|
|
|
|
/* 7th order interpolation (7 coefficients centered on 3rd) */
|
|
|
|
static fluid_real_t sinc_table7[FLUID_INTERP_MAX][7];
|
|
|
|
|
|
|
|
|
|
|
|
#define SINC_INTERP_ORDER 7 /* 7th order constant */
|
|
|
|
|
|
|
|
|
|
|
|
/* Initializes interpolation tables */
|
|
|
|
void fluid_rvoice_dsp_config (void)
|
|
|
|
{
|
|
|
|
int i, i2;
|
|
|
|
double x, v;
|
|
|
|
double i_shifted;
|
|
|
|
|
|
|
|
/* Initialize the coefficients for the interpolation. The math comes
|
|
|
|
* from a mail, posted by Olli Niemitalo to the music-dsp mailing
|
|
|
|
* list (I found it in the music-dsp archives
|
|
|
|
* http://www.smartelectronix.com/musicdsp/). */
|
|
|
|
|
|
|
|
for (i = 0; i < FLUID_INTERP_MAX; i++)
|
|
|
|
{
|
|
|
|
x = (double) i / (double) FLUID_INTERP_MAX;
|
|
|
|
|
|
|
|
interp_coeff[i][0] = (fluid_real_t)(x * (-0.5 + x * (1 - 0.5 * x)));
|
|
|
|
interp_coeff[i][1] = (fluid_real_t)(1.0 + x * x * (1.5 * x - 2.5));
|
|
|
|
interp_coeff[i][2] = (fluid_real_t)(x * (0.5 + x * (2.0 - 1.5 * x)));
|
|
|
|
interp_coeff[i][3] = (fluid_real_t)(0.5 * x * x * (x - 1.0));
|
|
|
|
|
|
|
|
interp_coeff_linear[i][0] = (fluid_real_t)(1.0 - x);
|
|
|
|
interp_coeff_linear[i][1] = (fluid_real_t)x;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* i: Offset in terms of whole samples */
|
|
|
|
for (i = 0; i < SINC_INTERP_ORDER; i++)
|
|
|
|
{ /* i2: Offset in terms of fractional samples ('subsamples') */
|
|
|
|
for (i2 = 0; i2 < FLUID_INTERP_MAX; i2++)
|
|
|
|
{
|
|
|
|
/* center on middle of table */
|
|
|
|
i_shifted = (double)i - ((double)SINC_INTERP_ORDER / 2.0)
|
|
|
|
+ (double)i2 / (double)FLUID_INTERP_MAX;
|
|
|
|
|
|
|
|
/* sinc(0) cannot be calculated straightforward (limit needed for 0/0) */
|
|
|
|
if (fabs (i_shifted) > 0.000001)
|
|
|
|
{
|
|
|
|
v = (fluid_real_t)sin (i_shifted * M_PI) / (M_PI * i_shifted);
|
|
|
|
/* Hamming window */
|
|
|
|
v *= (fluid_real_t)0.5 * (1.0 + cos (2.0 * M_PI * i_shifted / (fluid_real_t)SINC_INTERP_ORDER));
|
|
|
|
}
|
|
|
|
else v = 1.0;
|
|
|
|
|
|
|
|
sinc_table7[FLUID_INTERP_MAX - i2 - 1][i] = v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
for (i = 0; i < FLUID_INTERP_MAX; i++)
|
|
|
|
{
|
|
|
|
printf ("%d %0.3f %0.3f %0.3f %0.3f %0.3f %0.3f %0.3f\n",
|
|
|
|
i, sinc_table7[0][i], sinc_table7[1][i], sinc_table7[2][i],
|
|
|
|
sinc_table7[3][i], sinc_table7[4][i], sinc_table7[5][i], sinc_table7[6][i]);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
fluid_check_fpe("interpolation table calculation");
|
|
|
|
}
|
|
|
|
|
2017-12-12 20:02:21 +00:00
|
|
|
static FLUID_INLINE fluid_real_t
|
2018-01-18 16:13:50 +00:00
|
|
|
fluid_rvoice_get_float_sample(const short int* dsp_msb, const char* dsp_lsb, unsigned int idx)
|
2017-12-08 17:23:18 +00:00
|
|
|
{
|
2018-01-18 16:13:50 +00:00
|
|
|
int32_t sample = fluid_rvoice_get_sample(dsp_msb, dsp_lsb, idx);
|
|
|
|
return (fluid_real_t)sample;
|
2017-12-08 17:23:18 +00:00
|
|
|
}
|
|
|
|
|
2010-06-26 15:02:58 +00:00
|
|
|
/* No interpolation. Just take the sample, which is closest to
|
|
|
|
* the playback pointer. Questionable quality, but very
|
|
|
|
* efficient. */
|
|
|
|
int
|
2018-04-11 20:31:33 +00:00
|
|
|
fluid_rvoice_dsp_interpolate_none (fluid_rvoice_dsp_t *voice, fluid_real_t *FLUID_RESTRICT dsp_buf, int looping)
|
2010-06-26 15:02:58 +00:00
|
|
|
{
|
|
|
|
fluid_phase_t dsp_phase = voice->phase;
|
|
|
|
fluid_phase_t dsp_phase_incr;
|
|
|
|
short int *dsp_data = voice->sample->data;
|
2017-12-08 17:23:18 +00:00
|
|
|
char *dsp_data24 = voice->sample->data24;
|
2010-06-26 15:02:58 +00:00
|
|
|
fluid_real_t dsp_amp = voice->amp;
|
|
|
|
fluid_real_t dsp_amp_incr = voice->amp_incr;
|
|
|
|
unsigned int dsp_i = 0;
|
|
|
|
unsigned int dsp_phase_index;
|
|
|
|
unsigned int end_index;
|
|
|
|
|
|
|
|
/* Convert playback "speed" floating point value to phase index/fract */
|
|
|
|
fluid_phase_set_float (dsp_phase_incr, voice->phase_incr);
|
|
|
|
|
|
|
|
end_index = looping ? voice->loopend - 1 : voice->end;
|
|
|
|
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
dsp_phase_index = fluid_phase_index_round (dsp_phase); /* round to nearest point */
|
|
|
|
|
|
|
|
/* interpolate sequence of sample points */
|
|
|
|
for ( ; dsp_i < FLUID_BUFSIZE && dsp_phase_index <= end_index; dsp_i++)
|
|
|
|
{
|
2018-01-18 16:13:50 +00:00
|
|
|
dsp_buf[dsp_i] = dsp_amp * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index);
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index_round (dsp_phase); /* round to nearest point */
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* break out if not looping (buffer may not be full) */
|
|
|
|
if (!looping) break;
|
|
|
|
|
|
|
|
/* go back to loop start */
|
|
|
|
if (dsp_phase_index > end_index)
|
|
|
|
{
|
|
|
|
fluid_phase_sub_int (dsp_phase, voice->loopend - voice->loopstart);
|
|
|
|
voice->has_looped = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* break out if filled buffer */
|
|
|
|
if (dsp_i >= FLUID_BUFSIZE) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
voice->phase = dsp_phase;
|
|
|
|
voice->amp = dsp_amp;
|
|
|
|
|
|
|
|
return (dsp_i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Straight line interpolation.
|
|
|
|
* Returns number of samples processed (usually FLUID_BUFSIZE but could be
|
|
|
|
* smaller if end of sample occurs).
|
|
|
|
*/
|
|
|
|
int
|
2018-04-11 20:31:33 +00:00
|
|
|
fluid_rvoice_dsp_interpolate_linear (fluid_rvoice_dsp_t *voice, fluid_real_t *FLUID_RESTRICT dsp_buf, int looping)
|
2010-06-26 15:02:58 +00:00
|
|
|
{
|
|
|
|
fluid_phase_t dsp_phase = voice->phase;
|
|
|
|
fluid_phase_t dsp_phase_incr;
|
|
|
|
short int *dsp_data = voice->sample->data;
|
2017-12-08 17:23:18 +00:00
|
|
|
char *dsp_data24 = voice->sample->data24;
|
2010-06-26 15:02:58 +00:00
|
|
|
fluid_real_t dsp_amp = voice->amp;
|
|
|
|
fluid_real_t dsp_amp_incr = voice->amp_incr;
|
|
|
|
unsigned int dsp_i = 0;
|
|
|
|
unsigned int dsp_phase_index;
|
|
|
|
unsigned int end_index;
|
2017-12-12 20:02:21 +00:00
|
|
|
fluid_real_t point;
|
2018-04-11 20:14:18 +00:00
|
|
|
const fluid_real_t *FLUID_RESTRICT coeffs;
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* Convert playback "speed" floating point value to phase index/fract */
|
|
|
|
fluid_phase_set_float (dsp_phase_incr, voice->phase_incr);
|
|
|
|
|
|
|
|
/* last index before 2nd interpolation point must be specially handled */
|
|
|
|
end_index = (looping ? voice->loopend - 1 : voice->end) - 1;
|
|
|
|
|
|
|
|
/* 2nd interpolation point to use at end of loop or sample */
|
2018-01-18 16:13:50 +00:00
|
|
|
if (looping) point = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopstart); /* loop start */
|
|
|
|
else point = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->end); /* duplicate end for samples no longer looping */
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
|
|
|
|
/* interpolate the sequence of sample points */
|
|
|
|
for ( ; dsp_i < FLUID_BUFSIZE && dsp_phase_index <= end_index; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = interp_coeff_linear[fluid_phase_fract_to_tablerow (dsp_phase)];
|
2018-01-18 16:13:50 +00:00
|
|
|
dsp_buf[dsp_i] = dsp_amp * (coeffs[0] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
|
|
|
+ coeffs[1] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+1));
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* break out if buffer filled */
|
|
|
|
if (dsp_i >= FLUID_BUFSIZE) break;
|
|
|
|
|
|
|
|
end_index++; /* we're now interpolating the last point */
|
|
|
|
|
|
|
|
/* interpolate within last point */
|
|
|
|
for (; dsp_phase_index <= end_index && dsp_i < FLUID_BUFSIZE; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = interp_coeff_linear[fluid_phase_fract_to_tablerow (dsp_phase)];
|
2018-01-18 16:13:50 +00:00
|
|
|
dsp_buf[dsp_i] = dsp_amp * (coeffs[0] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
2017-12-08 17:23:18 +00:00
|
|
|
+ coeffs[1] * point);
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr; /* increment amplitude */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!looping) break; /* break out if not looping (end of sample) */
|
|
|
|
|
|
|
|
/* go back to loop start (if past */
|
|
|
|
if (dsp_phase_index > end_index)
|
|
|
|
{
|
|
|
|
fluid_phase_sub_int (dsp_phase, voice->loopend - voice->loopstart);
|
|
|
|
voice->has_looped = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* break out if filled buffer */
|
|
|
|
if (dsp_i >= FLUID_BUFSIZE) break;
|
|
|
|
|
|
|
|
end_index--; /* set end back to second to last sample point */
|
|
|
|
}
|
|
|
|
|
|
|
|
voice->phase = dsp_phase;
|
|
|
|
voice->amp = dsp_amp;
|
|
|
|
|
|
|
|
return (dsp_i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 4th order (cubic) interpolation.
|
|
|
|
* Returns number of samples processed (usually FLUID_BUFSIZE but could be
|
|
|
|
* smaller if end of sample occurs).
|
|
|
|
*/
|
|
|
|
int
|
2018-04-11 20:31:33 +00:00
|
|
|
fluid_rvoice_dsp_interpolate_4th_order (fluid_rvoice_dsp_t *voice, fluid_real_t *FLUID_RESTRICT dsp_buf, int looping)
|
2010-06-26 15:02:58 +00:00
|
|
|
{
|
|
|
|
fluid_phase_t dsp_phase = voice->phase;
|
|
|
|
fluid_phase_t dsp_phase_incr;
|
|
|
|
short int *dsp_data = voice->sample->data;
|
2017-12-08 17:23:18 +00:00
|
|
|
char *dsp_data24 = voice->sample->data24;
|
2010-06-26 15:02:58 +00:00
|
|
|
fluid_real_t dsp_amp = voice->amp;
|
|
|
|
fluid_real_t dsp_amp_incr = voice->amp_incr;
|
|
|
|
unsigned int dsp_i = 0;
|
|
|
|
unsigned int dsp_phase_index;
|
|
|
|
unsigned int start_index, end_index;
|
2017-12-12 20:02:21 +00:00
|
|
|
fluid_real_t start_point, end_point1, end_point2;
|
2018-04-11 20:14:18 +00:00
|
|
|
const fluid_real_t *FLUID_RESTRICT coeffs;
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* Convert playback "speed" floating point value to phase index/fract */
|
|
|
|
fluid_phase_set_float (dsp_phase_incr, voice->phase_incr);
|
|
|
|
|
|
|
|
/* last index before 4th interpolation point must be specially handled */
|
|
|
|
end_index = (looping ? voice->loopend - 1 : voice->end) - 2;
|
|
|
|
|
|
|
|
if (voice->has_looped) /* set start_index and start point if looped or not */
|
|
|
|
{
|
|
|
|
start_index = voice->loopstart;
|
2018-01-18 16:13:50 +00:00
|
|
|
start_point = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopend - 1); /* last point in loop (wrap around) */
|
2010-06-26 15:02:58 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
start_index = voice->start;
|
2018-01-18 16:13:50 +00:00
|
|
|
start_point = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->start); /* just duplicate the point */
|
2010-06-26 15:02:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* get points off the end (loop start if looping, duplicate point if end) */
|
|
|
|
if (looping)
|
|
|
|
{
|
2018-01-18 16:13:50 +00:00
|
|
|
end_point1 = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopstart);
|
|
|
|
end_point2 = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopstart + 1);
|
2010-06-26 15:02:58 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-01-18 16:13:50 +00:00
|
|
|
end_point1 = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->end);
|
2010-06-26 15:02:58 +00:00
|
|
|
end_point2 = end_point1;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
|
|
|
|
/* interpolate first sample point (start or loop start) if needed */
|
|
|
|
for ( ; dsp_phase_index == start_index && dsp_i < FLUID_BUFSIZE; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = interp_coeff[fluid_phase_fract_to_tablerow (dsp_phase)];
|
2017-12-08 17:23:18 +00:00
|
|
|
dsp_buf[dsp_i] = dsp_amp *
|
|
|
|
( coeffs[0] * start_point
|
2018-01-18 16:13:50 +00:00
|
|
|
+ coeffs[1] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
|
|
|
+ coeffs[2] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+1)
|
|
|
|
+ coeffs[3] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+2));
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* interpolate the sequence of sample points */
|
|
|
|
for ( ; dsp_i < FLUID_BUFSIZE && dsp_phase_index <= end_index; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = interp_coeff[fluid_phase_fract_to_tablerow (dsp_phase)];
|
2017-12-08 17:23:18 +00:00
|
|
|
dsp_buf[dsp_i] = dsp_amp *
|
2018-01-18 16:13:50 +00:00
|
|
|
( coeffs[0] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-1)
|
|
|
|
+ coeffs[1] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
|
|
|
+ coeffs[2] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+1)
|
|
|
|
+ coeffs[3] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+2));
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* break out if buffer filled */
|
|
|
|
if (dsp_i >= FLUID_BUFSIZE) break;
|
|
|
|
|
|
|
|
end_index++; /* we're now interpolating the 2nd to last point */
|
|
|
|
|
|
|
|
/* interpolate within 2nd to last point */
|
|
|
|
for (; dsp_phase_index <= end_index && dsp_i < FLUID_BUFSIZE; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = interp_coeff[fluid_phase_fract_to_tablerow (dsp_phase)];
|
2017-12-08 17:23:18 +00:00
|
|
|
dsp_buf[dsp_i] = dsp_amp *
|
2018-01-18 16:13:50 +00:00
|
|
|
( coeffs[0] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-1)
|
|
|
|
+ coeffs[1] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
|
|
|
+ coeffs[2] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+1)
|
2017-12-08 17:23:18 +00:00
|
|
|
+ coeffs[3] * end_point1);
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
end_index++; /* we're now interpolating the last point */
|
|
|
|
|
|
|
|
/* interpolate within the last point */
|
|
|
|
for (; dsp_phase_index <= end_index && dsp_i < FLUID_BUFSIZE; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = interp_coeff[fluid_phase_fract_to_tablerow (dsp_phase)];
|
2017-12-08 17:23:18 +00:00
|
|
|
dsp_buf[dsp_i] = dsp_amp *
|
2018-01-18 16:13:50 +00:00
|
|
|
( coeffs[0] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-1)
|
|
|
|
+ coeffs[1] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
2017-12-08 17:23:18 +00:00
|
|
|
+ coeffs[2] * end_point1
|
|
|
|
+ coeffs[3] * end_point2);
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!looping) break; /* break out if not looping (end of sample) */
|
|
|
|
|
|
|
|
/* go back to loop start */
|
|
|
|
if (dsp_phase_index > end_index)
|
|
|
|
{
|
|
|
|
fluid_phase_sub_int (dsp_phase, voice->loopend - voice->loopstart);
|
|
|
|
|
|
|
|
if (!voice->has_looped)
|
|
|
|
{
|
|
|
|
voice->has_looped = 1;
|
|
|
|
start_index = voice->loopstart;
|
2018-01-18 16:13:50 +00:00
|
|
|
start_point = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopend-1);
|
2010-06-26 15:02:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* break out if filled buffer */
|
|
|
|
if (dsp_i >= FLUID_BUFSIZE) break;
|
|
|
|
|
|
|
|
end_index -= 2; /* set end back to third to last sample point */
|
|
|
|
}
|
|
|
|
|
|
|
|
voice->phase = dsp_phase;
|
|
|
|
voice->amp = dsp_amp;
|
|
|
|
|
|
|
|
return (dsp_i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 7th order interpolation.
|
|
|
|
* Returns number of samples processed (usually FLUID_BUFSIZE but could be
|
|
|
|
* smaller if end of sample occurs).
|
|
|
|
*/
|
|
|
|
int
|
2018-04-11 20:31:33 +00:00
|
|
|
fluid_rvoice_dsp_interpolate_7th_order (fluid_rvoice_dsp_t *voice, fluid_real_t *FLUID_RESTRICT dsp_buf, int looping)
|
2010-06-26 15:02:58 +00:00
|
|
|
{
|
|
|
|
fluid_phase_t dsp_phase = voice->phase;
|
|
|
|
fluid_phase_t dsp_phase_incr;
|
|
|
|
short int *dsp_data = voice->sample->data;
|
2017-12-08 17:23:18 +00:00
|
|
|
char *dsp_data24 = voice->sample->data24;
|
2010-06-26 15:02:58 +00:00
|
|
|
fluid_real_t dsp_amp = voice->amp;
|
|
|
|
fluid_real_t dsp_amp_incr = voice->amp_incr;
|
|
|
|
unsigned int dsp_i = 0;
|
|
|
|
unsigned int dsp_phase_index;
|
|
|
|
unsigned int start_index, end_index;
|
2017-12-12 20:02:21 +00:00
|
|
|
fluid_real_t start_points[3], end_points[3];
|
2018-04-11 20:14:18 +00:00
|
|
|
const fluid_real_t *FLUID_RESTRICT coeffs;
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* Convert playback "speed" floating point value to phase index/fract */
|
|
|
|
fluid_phase_set_float (dsp_phase_incr, voice->phase_incr);
|
|
|
|
|
|
|
|
/* add 1/2 sample to dsp_phase since 7th order interpolation is centered on
|
|
|
|
* the 4th sample point */
|
|
|
|
fluid_phase_incr (dsp_phase, (fluid_phase_t)0x80000000);
|
|
|
|
|
|
|
|
/* last index before 7th interpolation point must be specially handled */
|
|
|
|
end_index = (looping ? voice->loopend - 1 : voice->end) - 3;
|
|
|
|
|
|
|
|
if (voice->has_looped) /* set start_index and start point if looped or not */
|
|
|
|
{
|
|
|
|
start_index = voice->loopstart;
|
2018-01-18 16:13:50 +00:00
|
|
|
start_points[0] = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopend - 1);
|
|
|
|
start_points[1] = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopend - 2);
|
|
|
|
start_points[2] = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopend - 3);
|
2010-06-26 15:02:58 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
start_index = voice->start;
|
2018-01-18 16:13:50 +00:00
|
|
|
start_points[0] = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->start); /* just duplicate the start point */
|
2010-06-26 15:02:58 +00:00
|
|
|
start_points[1] = start_points[0];
|
|
|
|
start_points[2] = start_points[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* get the 3 points off the end (loop start if looping, duplicate point if end) */
|
|
|
|
if (looping)
|
|
|
|
{
|
2018-01-18 16:13:50 +00:00
|
|
|
end_points[0] = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopstart);
|
|
|
|
end_points[1] = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopstart + 1);
|
|
|
|
end_points[2] = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopstart + 2);
|
2010-06-26 15:02:58 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-01-18 16:13:50 +00:00
|
|
|
end_points[0] = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->end);
|
2010-06-26 15:02:58 +00:00
|
|
|
end_points[1] = end_points[0];
|
|
|
|
end_points[2] = end_points[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
|
|
|
|
/* interpolate first sample point (start or loop start) if needed */
|
|
|
|
for ( ; dsp_phase_index == start_index && dsp_i < FLUID_BUFSIZE; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = sinc_table7[fluid_phase_fract_to_tablerow (dsp_phase)];
|
|
|
|
|
|
|
|
dsp_buf[dsp_i] = dsp_amp
|
2017-12-12 20:06:11 +00:00
|
|
|
* (coeffs[0] * start_points[2]
|
|
|
|
+ coeffs[1] * start_points[1]
|
|
|
|
+ coeffs[2] * start_points[0]
|
2018-01-18 16:13:50 +00:00
|
|
|
+ coeffs[3] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
|
|
|
+ coeffs[4] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+1)
|
|
|
|
+ coeffs[5] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+2)
|
|
|
|
+ coeffs[6] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+3));
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
start_index++;
|
|
|
|
|
|
|
|
/* interpolate 2nd to first sample point (start or loop start) if needed */
|
|
|
|
for ( ; dsp_phase_index == start_index && dsp_i < FLUID_BUFSIZE; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = sinc_table7[fluid_phase_fract_to_tablerow (dsp_phase)];
|
|
|
|
|
|
|
|
dsp_buf[dsp_i] = dsp_amp
|
2017-12-12 20:06:11 +00:00
|
|
|
* (coeffs[0] * start_points[1]
|
|
|
|
+ coeffs[1] * start_points[0]
|
2018-01-18 16:13:50 +00:00
|
|
|
+ coeffs[2] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-1)
|
|
|
|
+ coeffs[3] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
|
|
|
+ coeffs[4] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+1)
|
|
|
|
+ coeffs[5] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+2)
|
|
|
|
+ coeffs[6] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+3));
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
start_index++;
|
|
|
|
|
|
|
|
/* interpolate 3rd to first sample point (start or loop start) if needed */
|
|
|
|
for ( ; dsp_phase_index == start_index && dsp_i < FLUID_BUFSIZE; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = sinc_table7[fluid_phase_fract_to_tablerow (dsp_phase)];
|
|
|
|
|
|
|
|
dsp_buf[dsp_i] = dsp_amp
|
2017-12-12 20:06:11 +00:00
|
|
|
* (coeffs[0] * start_points[0]
|
2018-01-18 16:13:50 +00:00
|
|
|
+ coeffs[1] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-2)
|
|
|
|
+ coeffs[2] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-1)
|
|
|
|
+ coeffs[3] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
|
|
|
+ coeffs[4] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+1)
|
|
|
|
+ coeffs[5] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+2)
|
|
|
|
+ coeffs[6] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+3));
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
start_index -= 2; /* set back to original start index */
|
|
|
|
|
|
|
|
|
|
|
|
/* interpolate the sequence of sample points */
|
|
|
|
for ( ; dsp_i < FLUID_BUFSIZE && dsp_phase_index <= end_index; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = sinc_table7[fluid_phase_fract_to_tablerow (dsp_phase)];
|
|
|
|
|
|
|
|
dsp_buf[dsp_i] = dsp_amp
|
2018-01-18 16:13:50 +00:00
|
|
|
* (coeffs[0] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-3)
|
|
|
|
+ coeffs[1] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-2)
|
|
|
|
+ coeffs[2] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-1)
|
|
|
|
+ coeffs[3] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
|
|
|
+ coeffs[4] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+1)
|
|
|
|
+ coeffs[5] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+2)
|
|
|
|
+ coeffs[6] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+3));
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* break out if buffer filled */
|
|
|
|
if (dsp_i >= FLUID_BUFSIZE) break;
|
|
|
|
|
|
|
|
end_index++; /* we're now interpolating the 3rd to last point */
|
|
|
|
|
|
|
|
/* interpolate within 3rd to last point */
|
|
|
|
for (; dsp_phase_index <= end_index && dsp_i < FLUID_BUFSIZE; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = sinc_table7[fluid_phase_fract_to_tablerow (dsp_phase)];
|
|
|
|
|
|
|
|
dsp_buf[dsp_i] = dsp_amp
|
2018-01-18 16:13:50 +00:00
|
|
|
* (coeffs[0] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-3)
|
|
|
|
+ coeffs[1] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-2)
|
|
|
|
+ coeffs[2] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-1)
|
|
|
|
+ coeffs[3] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
|
|
|
+ coeffs[4] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+1)
|
|
|
|
+ coeffs[5] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+2)
|
2017-12-12 20:06:11 +00:00
|
|
|
+ coeffs[6] * end_points[0]);
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
end_index++; /* we're now interpolating the 2nd to last point */
|
|
|
|
|
|
|
|
/* interpolate within 2nd to last point */
|
|
|
|
for (; dsp_phase_index <= end_index && dsp_i < FLUID_BUFSIZE; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = sinc_table7[fluid_phase_fract_to_tablerow (dsp_phase)];
|
|
|
|
|
|
|
|
dsp_buf[dsp_i] = dsp_amp
|
2018-01-18 16:13:50 +00:00
|
|
|
* (coeffs[0] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-3)
|
|
|
|
+ coeffs[1] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-2)
|
|
|
|
+ coeffs[2] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-1)
|
|
|
|
+ coeffs[3] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
|
|
|
+ coeffs[4] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index+1)
|
2017-12-12 20:06:11 +00:00
|
|
|
+ coeffs[5] * end_points[0]
|
|
|
|
+ coeffs[6] * end_points[1]);
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
end_index++; /* we're now interpolating the last point */
|
|
|
|
|
|
|
|
/* interpolate within last point */
|
|
|
|
for (; dsp_phase_index <= end_index && dsp_i < FLUID_BUFSIZE; dsp_i++)
|
|
|
|
{
|
|
|
|
coeffs = sinc_table7[fluid_phase_fract_to_tablerow (dsp_phase)];
|
|
|
|
|
|
|
|
dsp_buf[dsp_i] = dsp_amp
|
2018-01-18 16:13:50 +00:00
|
|
|
* (coeffs[0] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-3)
|
|
|
|
+ coeffs[1] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-2)
|
|
|
|
+ coeffs[2] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index-1)
|
|
|
|
+ coeffs[3] * fluid_rvoice_get_float_sample(dsp_data, dsp_data24, dsp_phase_index)
|
2017-12-12 20:06:11 +00:00
|
|
|
+ coeffs[4] * end_points[0]
|
|
|
|
+ coeffs[5] * end_points[1]
|
|
|
|
+ coeffs[6] * end_points[2]);
|
2010-06-26 15:02:58 +00:00
|
|
|
|
|
|
|
/* increment phase and amplitude */
|
|
|
|
fluid_phase_incr (dsp_phase, dsp_phase_incr);
|
|
|
|
dsp_phase_index = fluid_phase_index (dsp_phase);
|
|
|
|
dsp_amp += dsp_amp_incr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!looping) break; /* break out if not looping (end of sample) */
|
|
|
|
|
|
|
|
/* go back to loop start */
|
|
|
|
if (dsp_phase_index > end_index)
|
|
|
|
{
|
|
|
|
fluid_phase_sub_int (dsp_phase, voice->loopend - voice->loopstart);
|
|
|
|
|
|
|
|
if (!voice->has_looped)
|
|
|
|
{
|
|
|
|
voice->has_looped = 1;
|
|
|
|
start_index = voice->loopstart;
|
2018-01-18 16:13:50 +00:00
|
|
|
start_points[0] = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopend - 1);
|
|
|
|
start_points[1] = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopend - 2);
|
|
|
|
start_points[2] = fluid_rvoice_get_float_sample(dsp_data, dsp_data24, voice->loopend - 3);
|
2010-06-26 15:02:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* break out if filled buffer */
|
|
|
|
if (dsp_i >= FLUID_BUFSIZE) break;
|
|
|
|
|
|
|
|
end_index -= 3; /* set end back to 4th to last sample point */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* sub 1/2 sample from dsp_phase since 7th order interpolation is centered on
|
|
|
|
* the 4th sample point (correct back to real value) */
|
|
|
|
fluid_phase_decr (dsp_phase, (fluid_phase_t)0x80000000);
|
|
|
|
|
|
|
|
voice->phase = dsp_phase;
|
|
|
|
voice->amp = dsp_amp;
|
|
|
|
|
|
|
|
return (dsp_i);
|
|
|
|
}
|