ESPHome 2026.6.0-dev
Loading...
Searching...
No Matches
i2s_audio_speaker.h
Go to the documentation of this file.
1#pragma once
2
3#ifdef USE_ESP32
4
5#include "../i2s_audio.h"
6
7#include <freertos/event_groups.h>
8#include <freertos/queue.h>
9#include <freertos/FreeRTOS.h>
10
14
16#include "esphome/core/gpio.h"
18
19namespace esphome::i2s_audio {
20
21// Shared constants used by both standard and SPDIF speaker implementations
22static constexpr size_t TASK_STACK_SIZE = 4096;
23static constexpr ssize_t TASK_PRIORITY = 19;
24
26 COMMAND_START = (1 << 0), // indicates loop should start speaker task
27 COMMAND_STOP = (1 << 1), // stops the speaker task
28 COMMAND_STOP_GRACEFULLY = (1 << 2), // Stops the speaker task once all data has been written
29
30 TASK_STARTING = (1 << 10),
31 TASK_RUNNING = (1 << 11),
32 TASK_STOPPING = (1 << 12),
33 TASK_STOPPED = (1 << 13),
34
35 ERR_ESP_NO_MEM = (1 << 19),
36
37 ERR_DROPPED_EVENT = (1 << 20), // ISR overflowed the event queue, dropping a completion event
38 ERR_PARTIAL_WRITE = (1 << 21), // i2s_channel_write returned fewer bytes than requested
39 ERR_LOCKSTEP_DESYNC = (1 << 22), // i2s_event_queue_ and write_records_queue_ fell out of sync
40
41 ALL_BITS = 0x00FFFFFF, // All valid FreeRTOS event group bits
42};
43
48 public:
49 float get_setup_priority() const override { return esphome::setup_priority::PROCESSOR; }
50
51 void setup() override;
52 void dump_config() override;
53 void loop() override;
54
55 void set_buffer_duration(uint32_t buffer_duration_ms) { this->buffer_duration_ms_ = buffer_duration_ms; }
56 void set_timeout(uint32_t ms) { this->timeout_ = ms; }
57 void set_dout_pin(uint8_t pin) { this->dout_pin_ = (gpio_num_t) pin; }
58
60 i2s_chan_handle_t get_tx_handle() const { return this->tx_handle_; }
61
62 void start() override;
63 void stop() override;
64 void finish() override;
65
66 void set_pause_state(bool pause_state) override { this->pause_state_ = pause_state; }
67 bool get_pause_state() const override { return this->pause_state_; }
68
75 size_t play(const uint8_t *data, size_t length, TickType_t ticks_to_wait) override;
76 size_t play(const uint8_t *data, size_t length) override { return play(data, length, 0); }
77
78 bool has_buffered_data() const override;
79
84 void set_volume(float volume) override;
85
90 void set_mute_state(bool mute_state) override;
91
92 protected:
95 static void speaker_task(void *params);
96
98 virtual void run_speaker_task() = 0;
99
102 void stop_(bool wait_on_empty);
103
109 static bool i2s_on_sent_cb(i2s_chan_handle_t handle, i2s_event_data_t *event, void *user_ctx);
110
114 virtual esp_err_t start_i2s_driver(audio::AudioStreamInfo &audio_stream_info) = 0;
115
122 esp_err_t init_i2s_channel_(const i2s_chan_config_t &chan_cfg, const i2s_std_config_t &std_cfg,
123 size_t event_queue_size);
124
126 void stop_i2s_driver_();
127
129 virtual void on_task_stopped() {}
130
134 void apply_software_volume_(uint8_t *data, size_t bytes_read);
135
140 void swap_esp32_mono_samples_(uint8_t *data, size_t bytes_read);
141
142 TaskHandle_t speaker_task_handle_{nullptr};
143 EventGroupHandle_t event_group_{nullptr};
144
145 // Lockstepped DMA buffer queues: i2s_event is outgoing, write_records is incoming
146 QueueHandle_t i2s_event_queue_{nullptr};
147 QueueHandle_t write_records_queue_{nullptr};
148
149 std::weak_ptr<ring_buffer::RingBuffer> audio_ring_buffer_;
150
152
153 optional<uint32_t> timeout_;
154
155 bool pause_state_{false};
156
157 int32_t q31_volume_factor_{INT32_MAX};
158
159 audio::AudioStreamInfo current_stream_info_; // The currently loaded driver's stream info
160
161 gpio_num_t dout_pin_;
162 i2s_chan_handle_t tx_handle_{nullptr};
163};
164
165} // namespace esphome::i2s_audio
166
167#endif // USE_ESP32
Abstract base class for I2S audio speaker implementations.
static bool i2s_on_sent_cb(i2s_chan_handle_t handle, i2s_event_data_t *event, void *user_ctx)
Callback function used to send playback timestamps to the speaker task.
void stop_i2s_driver_()
Stops the I2S driver and unlocks the I2S port.
void apply_software_volume_(uint8_t *data, size_t bytes_read)
Apply software volume control using Q15 fixed-point scaling.
std::weak_ptr< ring_buffer::RingBuffer > audio_ring_buffer_
void swap_esp32_mono_samples_(uint8_t *data, size_t bytes_read)
Swap adjacent 16-bit mono samples for ESP32 (non-variant) hardware quirk.
void set_buffer_duration(uint32_t buffer_duration_ms)
void set_pause_state(bool pause_state) override
virtual esp_err_t start_i2s_driver(audio::AudioStreamInfo &audio_stream_info)=0
Starts the ESP32 I2S driver.
esp_err_t init_i2s_channel_(const i2s_chan_config_t &chan_cfg, const i2s_std_config_t &std_cfg, size_t event_queue_size)
Shared I2S channel allocation, initialization, and event queue setup.
i2s_chan_handle_t get_tx_handle() const
Get the I2S TX channel handle.
virtual void on_task_stopped()
Called in loop() when the task has stopped. Override for mode-specific cleanup.
void stop_(bool wait_on_empty)
Plays the provided audio data.
virtual size_t play(const uint8_t *data, size_t length)=0
Plays the provided audio data.
virtual void set_volume(float volume)
Definition speaker.h:70
virtual void set_mute_state(bool mute_state)
Definition speaker.h:80
virtual bool has_buffered_data() const =0
__int64 ssize_t
Definition httplib.h:178
constexpr float PROCESSOR
For components that use data from sensors like displays.
Definition component.h:45
static void uint32_t
uint16_t length
Definition tt21100.cpp:0
spi_device_handle_t handle