Tasmota/lib/lib_audio/mp3_shine_esp32/src/l3subband.cpp
2024-04-17 11:28:17 +02:00

105 lines
4.1 KiB
C++
Executable File

/* L3SubBand */
#include "types.h"
#include "tables.h"
#include "l3subband.h"
/*
* shine_subband_initialise:
* ----------------------
* Calculates the analysis filterbank coefficients and rounds to the
* 9th decimal place accuracy of the filterbank tables in the ISO
* document. The coefficients are stored in #filter#
*/
void shine_subband_initialise(shine_global_config *config) {
int i,j;
double filter;
for(i=MAX_CHANNELS; i-- ; ) {
config->subband.off[i] = 0;
memset(config->subband.x[i], 0, sizeof(config->subband.x[i]));
}
for (i=SBLIMIT; i--; )
for (j=64; j--; )
{
if ((filter = 1e9*cos((double)((2*i+1)*(16-j)*PI64))) >= 0)
modf(filter+0.5, &filter);
else
modf(filter-0.5, &filter);
/* scale and convert to fixed point before storing */
config->subband.fl[i][j] = (int)(filter * (0x7fffffff * 1e-9));
}
}
/*
* shine_window_filter_subband:
* -------------------------
* Overlapping window on PCM samples
* 32 16-bit pcm samples are scaled to fractional 2's complement and
* concatenated to the end of the window buffer #x#. The updated window
* buffer #x# is then windowed by the analysis window #shine_enwindow# to produce
* the windowed sample #z#
* Calculates the analysis filter bank coefficients
* The windowed samples #z# is filtered by the digital filter matrix #filter#
* to produce the subband samples #s#. This done by first selectively
* picking out values from the windowed samples, and then multiplying
* them by the filter matrix, producing 32 subband samples.
*/
void shine_window_filter_subband(int16_t **buffer, int s[SBLIMIT], int ch, shine_global_config *config, int stride) {
int y[64];
int i,j;
int16_t *ptr = *buffer;
/* replace 32 oldest samples with 32 new samples */
for (i=32;i--;) {
config->subband.x[ch][i+config->subband.off[ch]] = ((int)*ptr) << 16;
ptr += stride;
}
*buffer = ptr;
for (i=64; i--; ) {
int s_value;
#ifdef __BORLANDC__
uint32_t s_value_lo;
#else
uint32_t s_value_lo __attribute__((unused));
#endif
asm_mul0 (s_value, s_value_lo, config->subband.x[ch][(config->subband.off[ch] + i + (0<<6)) & (HAN_SIZE-1)], shine_enwindow[i + (0<<6)]);
asm_muladd(s_value, s_value_lo, config->subband.x[ch][(config->subband.off[ch] + i + (1<<6)) & (HAN_SIZE-1)], shine_enwindow[i + (1<<6)]);
asm_muladd(s_value, s_value_lo, config->subband.x[ch][(config->subband.off[ch] + i + (2<<6)) & (HAN_SIZE-1)], shine_enwindow[i + (2<<6)]);
asm_muladd(s_value, s_value_lo, config->subband.x[ch][(config->subband.off[ch] + i + (3<<6)) & (HAN_SIZE-1)], shine_enwindow[i + (3<<6)]);
asm_muladd(s_value, s_value_lo, config->subband.x[ch][(config->subband.off[ch] + i + (4<<6)) & (HAN_SIZE-1)], shine_enwindow[i + (4<<6)]);
asm_muladd(s_value, s_value_lo, config->subband.x[ch][(config->subband.off[ch] + i + (5<<6)) & (HAN_SIZE-1)], shine_enwindow[i + (5<<6)]);
asm_muladd(s_value, s_value_lo, config->subband.x[ch][(config->subband.off[ch] + i + (6<<6)) & (HAN_SIZE-1)], shine_enwindow[i + (6<<6)]);
asm_muladd(s_value, s_value_lo, config->subband.x[ch][(config->subband.off[ch] + i + (7<<6)) & (HAN_SIZE-1)], shine_enwindow[i + (7<<6)]);
asm_mulz (s_value, s_value_lo);
y[i] = s_value;
}
config->subband.off[ch] = (config->subband.off[ch] + 480) & (HAN_SIZE-1); /* offset is modulo (HAN_SIZE)*/
for (i=SBLIMIT; i--; ) {
int s_value;
#ifdef __BORLANDC__
uint32_t s_value_lo;
#else
uint32_t s_value_lo __attribute__((unused));
#endif
asm_mul0(s_value, s_value_lo, config->subband.fl[i][63], y[63]);
for (j=63; j; j-=7) {
asm_muladd(s_value, s_value_lo, config->subband.fl[i][j-1], y[j-1]);
asm_muladd(s_value, s_value_lo, config->subband.fl[i][j-2], y[j-2]);
asm_muladd(s_value, s_value_lo, config->subband.fl[i][j-3], y[j-3]);
asm_muladd(s_value, s_value_lo, config->subband.fl[i][j-4], y[j-4]);
asm_muladd(s_value, s_value_lo, config->subband.fl[i][j-5], y[j-5]);
asm_muladd(s_value, s_value_lo, config->subband.fl[i][j-6], y[j-6]);
asm_muladd(s_value, s_value_lo, config->subband.fl[i][j-7], y[j-7]);
}
asm_mulz(s_value, s_value_lo);
s[i] = s_value;
}
}