aboutsummaryrefslogtreecommitdiffstats
path: root/main.cpp
blob: 5151e336dafd43027f6b9181736ef210e9c8eb1b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
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);
    }
}