mirror of
https://github.com/esphome/esphome.git
synced 2025-09-17 02:32:20 +01:00
197 lines
6.4 KiB
C++
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
|