diff options
Diffstat (limited to 'main.cpp')
-rw-r--r-- | main.cpp | 124 |
1 files changed, 124 insertions, 0 deletions
diff --git a/main.cpp b/main.cpp new file mode 100644 index 0000000..5151e33 --- /dev/null +++ b/main.cpp @@ -0,0 +1,124 @@ +#include "hal.h" +#include "ch.h" +#include "sos-iir-filter.h" + +#include <algorithm> +#include <array> +#include <cstring> +#include <ranges> + +static constexpr auto& WEIGHTING = A_weighting; +static constexpr auto& MIC_EQUALIZER = SPH0645LM4H_B_RB; +static constexpr sos_t MIC_OFFSET_DB ( 0.f); // Linear offset +static constexpr sos_t MIC_SENSITIVITY (-26.f); // dBFS value expected at MIC_REF_DB +static constexpr sos_t MIC_REF_DB ( 94.f); // dB where sensitivity is specified +static constexpr sos_t MIC_OVERLOAD_DB (120.f); // dB - Acoustic overload point +static constexpr sos_t MIC_NOISE_DB ( 29.f); // dB - Noise floor +static constexpr auto MIC_BITS = 16u; // SPH0645 is actually 18 bits +static constexpr auto SAMPLE_RATE = 32000u; + +static constexpr unsigned I2S_BUFSIZ = 512; // was 1024 +static constexpr unsigned I2S_STRIDE = 16; // was 8 + +// Calculate reference amplitude value at compile time +static const auto MIC_REF_AMPL = fpm::pow(sos_t(10), MIC_SENSITIVITY / 20) * + ((1 << (MIC_BITS - 1)) - 1); + +static SEMAPHORE_DECL(i2sReady, 0); +static THD_WORKING_AREA(waThread1, 128); +static std::array<uint32_t, I2S_BUFSIZ> i2sBuffer; +static sos_t Leq_sum_sqr (0); +static unsigned Leq_samples = 0; + +static THD_FUNCTION(Thread1, arg); +static void i2sCallback(I2SDriver *i2s); + +static constexpr auto I2SPRval = SAMPLE_RATE * 64 / 1000 / 1024 * 2; +static constexpr I2SConfig i2sConfig = { + /* TX buffer */ NULL, + /* RX buffer */ i2sBuffer.data(), + /* Size */ i2sBuffer.size(), + /* Callback */ i2sCallback, + /* I2SCFGR */ (3 << SPI_I2SCFGR_I2SCFG_Pos) | // Master receive + (0 << SPI_I2SCFGR_I2SSTD_Pos) | // Philips I2S + (1 << SPI_I2SCFGR_DATLEN_Pos) | // 24-bit + SPI_I2SCFGR_CHLEN, // 32-bit frame + /* I2SPR */ I2SPRval | (((I2SPRval / 2) & 1) ? SPI_I2SPR_ODD : 0) +}; + +THD_TABLE_BEGIN + THD_TABLE_THREAD(0, "blinker1", waThread1, Thread1, NULL) +THD_TABLE_END + +int main(void) +{ + halInit(); + chSysInit(); + for (;;) + asm("wfi"); +} + +THD_FUNCTION(Thread1, arg) +{ + (void)arg; + + chThdSleepMilliseconds(2000); + + palSetPadMode(GPIOF, 2, PAL_MODE_UNCONNECTED); + palSetLineMode(LINE_I2S_SD, PAL_MODE_ALTERNATE(0)); + palSetLineMode(LINE_I2S_WS, PAL_MODE_ALTERNATE(0)); + palSetLineMode(LINE_I2S_CK, PAL_MODE_ALTERNATE(0)); + palSetLineMode(LINE_USART2_TX, PAL_MODE_ALTERNATE(1)); + + sdStart(&SD2, NULL); + sdWrite(&SD2, (uint8_t *)"Noisemeter\n", 11); + chThdSleepMilliseconds(100); + + i2sStart(&I2SD1, &i2sConfig); + i2sStartExchange(&I2SD1); + + uint8_t strbuf[7] = { 0, 0, 0, 'd', 'B', '\n', '\0' }; + for (;;) { + chSemWait(&i2sReady); + + const auto Leq_RMS = fpm::sqrt(Leq_sum_sqr / Leq_samples); + const auto Leq_dB = MIC_OFFSET_DB + MIC_REF_DB + 20 * + fpm::log10(Leq_RMS / MIC_REF_AMPL); + Leq_sum_sqr = sos_t(0); + Leq_samples = 0; + + auto n = std::clamp(static_cast<int32_t>(Leq_dB), 0l, 999l); + strbuf[2] = n % 10 + '0'; n /= 10; + strbuf[1] = n % 10 + '0'; n /= 10; + strbuf[0] = n ? n + '0' : ' '; + sdWrite(&SD2, strbuf, sizeof(strbuf)); + } +} + +void i2sCallback(I2SDriver *i2s) +{ + // Note: samples come in as 16-bit big-endian, so for simplicity we just + // take the "high" 16-bits of valid data and drop the rest. Mic is + // technically 18-bit, so we are losing some precision. + + const auto halfsize = i2sBuffer.size() / 2; + const auto offset = i2sIsBufferComplete(i2s) ? halfsize : 0; + auto samples = reinterpret_cast<sos_t *>(i2sBuffer.data() + offset); + std::ranges::copy( + std::views::counted(i2sBuffer.begin() + offset, halfsize) + | std::ranges::views::stride(2 * I2S_STRIDE) + | std::views::transform([](auto s) { return sos_t((int16_t)s); }), + samples); + auto samps = std::views::counted(samples, halfsize / (2 * I2S_STRIDE)); + + // Accumulate Leq sum + MIC_EQUALIZER.filter(samps); + Leq_sum_sqr += WEIGHTING.filter_sum_sqr(samps); + Leq_samples += samps.size(); + + // Wakeup main thread for dB calculation every second + if (Leq_samples >= 2 * SAMPLE_RATE / I2S_STRIDE) { + chSemSignalI(&i2sReady); + } +} + |