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

◆ fastDb()

static float fl::audio::fft::Context::fastDb ( u32 x)
inlinestaticprivate

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

350 {
351 if (x == 0) return 0.0f;
352
353 // Find highest set bit (integer part of log2)
354 int msb = 0;
355 {
356 u32 v = x;
357 if (v >= 0x10000u) { v >>= 16; msb += 16; }
358 if (v >= 0x100u) { v >>= 8; msb += 8; }
359 if (v >= 0x10u) { v >>= 4; msb += 4; }
360 if (v >= 0x4u) { v >>= 2; msb += 2; }
361 if (v >= 0x2u) { msb += 1; }
362 }
363
364 // Normalized mantissa in [0, 1) as u16x16
365 u32 t_raw;
366 if (msb >= 16) {
367 t_raw = (x >> (msb - 16)) - 65536u;
368 } else {
369 t_raw = (x << (16 - msb)) - 65536u;
370 }
371 u16x16 t = u16x16::from_raw(t_raw);
372
373 // log2(1+t) ~= t + 0.345 * t * (1-t) [max error ~0.008]
374 // 22610 = 0.345 * 65536 (from_raw preserves exact original constant)
375 static constexpr u16x16 one(1.0f);
376 static constexpr u16x16 kCorrection = u16x16::from_raw(22610u);
377 u16x16 complement = one - t;
378 u16x16 prod = t * complement;
379 u16x16 correction = prod * kCorrection;
380 u16x16 frac = t + correction;
381
382 // log2(x) = msb + frac, assembled as u16x16
383 u16x16 log2_val = u16x16::from_raw(
384 (static_cast<u32>(msb) << 16) + frac.raw());
385
386 // 20*log10(x) = 6.02060 * log2(x)
387#if FASTLED_FFT_PRECISION == FASTLED_FFT_FIXED16
388 // 394593 = 6.02060 * 65536 (from_raw preserves exact original constant)
389 static constexpr u16x16 kDbScale = u16x16::from_raw(394593u);
390 u16x16 db = log2_val * kDbScale;
391 return db.to_float();
392#else
393 return 6.02060f * log2_val.to_float();
394#endif
395 }
static constexpr FASTLED_FORCE_INLINE u16x16 from_raw(u32 raw) FL_NOEXCEPT
Definition u16x16.h:53

References fl::u16x16::from_raw(), fl::u16x16::raw(), fl::t, fl::u16x16::to_float(), and fl::x.

+ Here is the call graph for this function: