1
0
mirror of https://github.com/esphome/esphome.git synced 2025-09-17 02:32:20 +01:00
Files
esphome/esphome/components/sound_level/sound_level.cpp
2025-06-09 00:02:30 +00:00

197 lines
6.4 KiB
C++

#include "sound_level.h"
#ifdef USE_ESP32
#include "esphome/core/log.h"
#include <cmath>
#include <cstdint>
namespace esphome {
namespace sound_level {
static const char *const TAG = "sound_level";
static const uint32_t AUDIO_BUFFER_DURATION_MS = 30;
static const uint32_t RING_BUFFER_DURATION_MS = 120;
// Square INT16_MIN since INT16_MIN^2 > INT16_MAX^2
static const double MAX_SAMPLE_SQUARED_DENOMINATOR = INT16_MIN * INT16_MIN;
void SoundLevelComponent::dump_config() {
ESP_LOGCONFIG(TAG,
"Sound Level Component:\n"
" Measurement Duration: %" PRIu32 " ms",
measurement_duration_ms_);
LOG_SENSOR(" ", "Peak:", this->peak_sensor_);
LOG_SENSOR(" ", "RMS:", this->rms_sensor_);
}
void SoundLevelComponent::setup() {
this->microphone_source_->add_data_callback([this](const std::vector<uint8_t> &data) {
std::shared_ptr<RingBuffer> temp_ring_buffer = this->ring_buffer_.lock();
if (this->ring_buffer_.use_count() == 2) {
// ``audio_buffer_`` and ``temp_ring_buffer`` share ownership of a ring buffer, so its safe/useful to write
temp_ring_buffer->write((void *) data.data(), data.size());
}
});
if (!this->microphone_source_->is_passive()) {
// Automatically start the microphone if not in passive mode
this->microphone_source_->start();
}
}
void SoundLevelComponent::loop() {
if ((this->peak_sensor_ == nullptr) && (this->rms_sensor_ == nullptr)) {
// No sensors configured, nothing to do
return;
}
if (this->microphone_source_->is_running() && !this->status_has_error()) {
// Allocate buffers
if (this->start_()) {
this->status_clear_warning();
}
} else {
if (!this->status_has_warning()) {
this->status_set_warning("Microphone isn't running, can't compute statistics");
// Deallocate buffers, if necessary
this->stop_();
// Reset sensor outputs
if (this->peak_sensor_ != nullptr) {
this->peak_sensor_->publish_state(NAN);
}
if (this->rms_sensor_ != nullptr) {
this->rms_sensor_->publish_state(NAN);
}
// Reset accumulators
this->squared_peak_ = 0;
this->squared_samples_sum_ = 0;
this->sample_count_ = 0;
}
return;
}
if (this->status_has_error()) {
return;
}
// Copy data from ring buffer into the transfer buffer - don't block to avoid slowing the main loop
this->audio_buffer_->transfer_data_from_source(0);
if (this->audio_buffer_->available() == 0) {
// No new audio available for processing
return;
}
const uint32_t samples_in_window =
this->microphone_source_->get_audio_stream_info().ms_to_samples(this->measurement_duration_ms_);
const uint32_t samples_available_to_process =
this->microphone_source_->get_audio_stream_info().bytes_to_samples(this->audio_buffer_->available());
const uint32_t samples_to_process = std::min(samples_in_window - this->sample_count_, samples_available_to_process);
// MicrophoneSource always provides int16 samples due to Python codegen settings
const int16_t *audio_data = reinterpret_cast<const int16_t *>(this->audio_buffer_->get_buffer_start());
// Process all the new audio samples
for (uint32_t i = 0; i < samples_to_process; ++i) {
// Squaring int16 samples won't overflow an int32
int32_t squared_sample = static_cast<int32_t>(audio_data[i]) * static_cast<int32_t>(audio_data[i]);
if (this->peak_sensor_ != nullptr) {
this->squared_peak_ = std::max(this->squared_peak_, squared_sample);
}
if (this->rms_sensor_ != nullptr) {
// Squared sum is an uint64 type - at max levels, an uint32 type would overflow after ~8 samples
this->squared_samples_sum_ += squared_sample;
}
++this->sample_count_;
}
// Remove the processed samples from ``audio_buffer_``
this->audio_buffer_->decrease_buffer_length(
this->microphone_source_->get_audio_stream_info().samples_to_bytes(samples_to_process));
if (this->sample_count_ == samples_in_window) {
// Processed enough samples for the measurement window, compute and publish the sensor values
if (this->peak_sensor_ != nullptr) {
const float peak_db = 10.0f * log10(static_cast<float>(this->squared_peak_) / MAX_SAMPLE_SQUARED_DENOMINATOR);
this->peak_sensor_->publish_state(peak_db);
this->squared_peak_ = 0; // reset accumulator
}
if (this->rms_sensor_ != nullptr) {
// Calculations are done with doubles instead of floats - floats lose precision for even modest window durations
const double rms_db = 10.0 * log10((this->squared_samples_sum_ / MAX_SAMPLE_SQUARED_DENOMINATOR) /
static_cast<double>(samples_in_window));
this->rms_sensor_->publish_state(rms_db);
this->squared_samples_sum_ = 0; // reset accumulator
}
this->sample_count_ = 0; // reset counter
}
}
void SoundLevelComponent::start() {
if (this->microphone_source_->is_passive()) {
ESP_LOGW(TAG, "Can't start the microphone in passive mode");
return;
}
this->microphone_source_->start();
}
void SoundLevelComponent::stop() {
if (this->microphone_source_->is_passive()) {
ESP_LOGW(TAG, "Can't stop microphone in passive mode");
return;
}
this->microphone_source_->stop();
}
bool SoundLevelComponent::start_() {
if (this->audio_buffer_ != nullptr) {
return true;
}
// Allocate a transfer buffer
this->audio_buffer_ = audio::AudioSourceTransferBuffer::create(
this->microphone_source_->get_audio_stream_info().ms_to_bytes(AUDIO_BUFFER_DURATION_MS));
if (this->audio_buffer_ == nullptr) {
this->status_momentary_error("Failed to allocate transfer buffer", 15000);
return false;
}
// Allocates a new ring buffer, adds it as a source for the transfer buffer, and points ring_buffer_ to it
this->ring_buffer_.reset(); // Reset pointer to any previous ring buffer allocation
std::shared_ptr<RingBuffer> temp_ring_buffer =
RingBuffer::create(this->microphone_source_->get_audio_stream_info().ms_to_bytes(RING_BUFFER_DURATION_MS));
if (temp_ring_buffer.use_count() == 0) {
this->status_momentary_error("Failed to allocate ring buffer", 15000);
this->stop_();
return false;
} else {
this->ring_buffer_ = temp_ring_buffer;
this->audio_buffer_->set_source(temp_ring_buffer);
}
this->status_clear_error();
return true;
}
void SoundLevelComponent::stop_() { this->audio_buffer_.reset(); }
} // namespace sound_level
} // namespace esphome
#endif