aboutsummaryrefslogtreecommitdiffstats
path: root/main.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'main.cpp')
-rw-r--r--main.cpp124
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);
+ }
+}
+