FastLED 3.9.15
Loading...
Searching...
No Matches

◆ runLogRebin()

void fl::audio::fft::Context::runLogRebin ( span< const i16 > buffer,
Bins * out )
inlineprivate

Definition at line 237 of file fft_impl.cpp.hpp.

237 {
238 out->setParams(mFmin, mFmax, mSampleRate);
239 const int N = mInputSamples;
240 const int bands = mTotalBands;
241 const int numRawBins = N / 2 + 1;
242
243 // Use thread-local scratch buffers to avoid stack overflow on
244 // ESP32 tasks with limited stack (~4-8 KB).
245 FftScratch &s = scratch();
246 s.windowed.resize(N);
247 s.fftOut.resize(N);
248 s.re.resize(numRawBins);
249 s.im.resize(numRawBins);
250 s.mag.resize(numRawBins);
251 s.rawBinsI.resize(bands);
252
253 applyWindow(buffer.data(), mWindowBuf.data(), s.windowed.data(), N);
254 fl_fft_real_forward(mFftrCfg, mInputSamples, s.windowed.data(), s.fftOut.data());
255
256 // Deinterleave AoS → SoA and batch-compute magnitudes
257 deinterleave(s.fftOut.data(), s.re.data(), s.im.data(), numRawBins);
258 batchMag(s.re.data(), s.im.data(), s.mag.data(), numRawBins);
259
260 // Linear bins (same as other paths)
261 computeLinearBins(s.mag.data(), N, out);
262
263 // Group FFT bins into log-spaced output bins (integer accumulation)
264 fl::memset(s.rawBinsI.data(), 0, sizeof(u32) * bands);
265 logRebinRange(s.mag.data(), N, static_cast<float>(mSampleRate),
266 0, bands, s.rawBinsI.data(), mLogBinLut);
267
268 // Store raw magnitudes (dB computed lazily by Bins::db())
269 fl::vector<float> &rawBins = out->raw_mut();
270 rawBins.resize(bands);
271 for (int i = 0; i < bands; ++i) {
272 rawBins[i] = static_cast<float>(s.rawBinsI[i]);
273 }
274
275 // Store bin-width normalization factors so consumers can optionally
276 // normalize (e.g. for equalization display). Raw output is unchanged.
277 out->setNormFactors(mLogBinNormFactors);
278 }
static void applyWindow(const kiss_fft_scalar *samples, const alpha16 *win, kiss_fft_scalar *out, int N)
static void deinterleave(const kiss_fft_cpx *cpx, kiss_fft_scalar *re, kiss_fft_scalar *im, int n)
fl::vector< alpha16 > mWindowBuf
static void batchMag(const kiss_fft_scalar *re, const kiss_fft_scalar *im, u16 *mag, int n)
static FftScratch & scratch()
fl::vector< u8 > mLogBinLut
void logRebinRange(const u16 *mag, int fftN, float fs, int binStart, int binEnd, u32 *rawBinsI, const fl::vector< u8 > &lut)
fl::vector< float > mLogBinNormFactors
void computeLinearBins(const u16 *mag, int, Bins *out)
void resize(fl::size n) FL_NOEXCEPT
Definition vector.h:593
void fl_fft_real_forward(kiss_fftr_cfg cfg, int N, const kiss_fft_scalar *in, kiss_fft_cpx *out) FL_NOEXCEPT
Forward real-to-complex FFT.
void * memset(void *s, int c, size_t n) FL_NOEXCEPT

References applyWindow(), batchMag(), computeLinearBins(), fl::span< T, Extent >::data(), fl::vector< T >::data(), deinterleave(), fl::audio::fft::Context::FftScratch::fftOut, fl::audio::fft::fl_fft_real_forward(), fl::audio::fft::Context::FftScratch::im, logRebinRange(), fl::audio::fft::Context::FftScratch::mag, fl::memset(), mFftrCfg, mFmax, mFmin, mInputSamples, mLogBinLut, mLogBinNormFactors, mSampleRate, mTotalBands, mWindowBuf, fl::audio::fft::Bins::raw_mut(), fl::audio::fft::Context::FftScratch::rawBinsI, fl::audio::fft::Context::FftScratch::re, fl::vector< T >::resize(), scratch(), fl::audio::fft::Bins::setNormFactors(), fl::audio::fft::Bins::setParams(), and fl::audio::fft::Context::FftScratch::windowed.

Referenced by run().

+ Here is the call graph for this function:
+ Here is the caller graph for this function: