mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-25 05:03:52 +01:00 
			
		
		
		
	esp32 remote: make RMT memory blocks configureable (#1002)
This commit is contained in:
		| @@ -6,14 +6,24 @@ namespace remote_base { | |||||||
|  |  | ||||||
| static const char *TAG = "remote_base"; | static const char *TAG = "remote_base"; | ||||||
|  |  | ||||||
| RemoteComponentBase::RemoteComponentBase(GPIOPin *pin) : pin_(pin) { |  | ||||||
| #ifdef ARDUINO_ARCH_ESP32 | #ifdef ARDUINO_ARCH_ESP32 | ||||||
|  | RemoteRMTChannel::RemoteRMTChannel(uint8_t mem_block_num) : mem_block_num_(mem_block_num) { | ||||||
|   static rmt_channel_t next_rmt_channel = RMT_CHANNEL_0; |   static rmt_channel_t next_rmt_channel = RMT_CHANNEL_0; | ||||||
|   this->channel_ = next_rmt_channel; |   this->channel_ = next_rmt_channel; | ||||||
|   next_rmt_channel = rmt_channel_t(int(next_rmt_channel) + 1); |   next_rmt_channel = rmt_channel_t(int(next_rmt_channel) + mem_block_num); | ||||||
| #endif |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | void RemoteRMTChannel::config_rmt(rmt_config_t &rmt) { | ||||||
|  |   if (rmt_channel_t(int(this->channel_) + this->mem_block_num_) > RMT_CHANNEL_7) { | ||||||
|  |     this->mem_block_num_ = int(RMT_CHANNEL_7) - int(this->channel_) + 1; | ||||||
|  |     ESP_LOGW(TAG, "Not enough RMT memory blocks avaiable, reduced to %i blocks.", this->mem_block_num_); | ||||||
|  |   } | ||||||
|  |   rmt.channel = this->channel_; | ||||||
|  |   rmt.clk_div = this->clock_divider_; | ||||||
|  |   rmt.mem_block_num = this->mem_block_num_; | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  |  | ||||||
| void RemoteReceiverBinarySensorBase::dump_config() { LOG_BINARY_SENSOR("", "Remote Receiver Binary Sensor", this); } | void RemoteReceiverBinarySensorBase::dump_config() { LOG_BINARY_SENSOR("", "Remote Receiver Binary Sensor", this); } | ||||||
|  |  | ||||||
| void RemoteTransmitterBase::send_(uint32_t send_times, uint32_t send_wait) { | void RemoteTransmitterBase::send_(uint32_t send_times, uint32_t send_wait) { | ||||||
|   | |||||||
| @@ -146,15 +146,21 @@ template<typename T> class RemoteProtocol { | |||||||
|  |  | ||||||
| class RemoteComponentBase { | class RemoteComponentBase { | ||||||
|  public: |  public: | ||||||
|   explicit RemoteComponentBase(GPIOPin *pin); |   explicit RemoteComponentBase(GPIOPin *pin) : pin_(pin){}; | ||||||
|  |  | ||||||
| #ifdef ARDUINO_ARCH_ESP32 |  | ||||||
|   void set_channel(rmt_channel_t channel) { this->channel_ = channel; } |  | ||||||
|   void set_clock_divider(uint8_t clock_divider) { this->clock_divider_ = clock_divider; } |  | ||||||
| #endif |  | ||||||
|  |  | ||||||
|  protected: |  protected: | ||||||
|  |   GPIOPin *pin_; | ||||||
|  | }; | ||||||
|  |  | ||||||
| #ifdef ARDUINO_ARCH_ESP32 | #ifdef ARDUINO_ARCH_ESP32 | ||||||
|  | class RemoteRMTChannel { | ||||||
|  |  public: | ||||||
|  |   explicit RemoteRMTChannel(uint8_t mem_block_num = 1); | ||||||
|  |  | ||||||
|  |   void config_rmt(rmt_config_t &rmt); | ||||||
|  |   void set_clock_divider(uint8_t clock_divider) { this->clock_divider_ = clock_divider; } | ||||||
|  |  | ||||||
|  |  protected: | ||||||
|   uint32_t from_microseconds(uint32_t us) { |   uint32_t from_microseconds(uint32_t us) { | ||||||
|     const uint32_t ticks_per_ten_us = 80000000u / this->clock_divider_ / 100000u; |     const uint32_t ticks_per_ten_us = 80000000u / this->clock_divider_ / 100000u; | ||||||
|     return us * ticks_per_ten_us / 10; |     return us * ticks_per_ten_us / 10; | ||||||
| @@ -163,15 +169,12 @@ class RemoteComponentBase { | |||||||
|     const uint32_t ticks_per_ten_us = 80000000u / this->clock_divider_ / 100000u; |     const uint32_t ticks_per_ten_us = 80000000u / this->clock_divider_ / 100000u; | ||||||
|     return (ticks * 10) / ticks_per_ten_us; |     return (ticks * 10) / ticks_per_ten_us; | ||||||
|   } |   } | ||||||
| #endif |   RemoteComponentBase *remote_base_; | ||||||
|  |  | ||||||
|   GPIOPin *pin_; |  | ||||||
| #ifdef ARDUINO_ARCH_ESP32 |  | ||||||
|   rmt_channel_t channel_{RMT_CHANNEL_0}; |   rmt_channel_t channel_{RMT_CHANNEL_0}; | ||||||
|  |   uint8_t mem_block_num_; | ||||||
|   uint8_t clock_divider_{80}; |   uint8_t clock_divider_{80}; | ||||||
|   esp_err_t error_code_{ESP_OK}; |  | ||||||
| #endif |  | ||||||
| }; | }; | ||||||
|  | #endif | ||||||
|  |  | ||||||
| class RemoteTransmitterBase : public RemoteComponentBase { | class RemoteTransmitterBase : public RemoteComponentBase { | ||||||
|  public: |  public: | ||||||
|   | |||||||
| @@ -3,7 +3,8 @@ import esphome.config_validation as cv | |||||||
| from esphome import pins | from esphome import pins | ||||||
| from esphome.components import remote_base | from esphome.components import remote_base | ||||||
| from esphome.const import CONF_BUFFER_SIZE, CONF_DUMP, CONF_FILTER, CONF_ID, CONF_IDLE, \ | from esphome.const import CONF_BUFFER_SIZE, CONF_DUMP, CONF_FILTER, CONF_ID, CONF_IDLE, \ | ||||||
|     CONF_PIN, CONF_TOLERANCE |     CONF_PIN, CONF_TOLERANCE, CONF_MEMORY_BLOCKS | ||||||
|  | from esphome.core import CORE | ||||||
|  |  | ||||||
| AUTO_LOAD = ['remote_base'] | AUTO_LOAD = ['remote_base'] | ||||||
| remote_receiver_ns = cg.esphome_ns.namespace('remote_receiver') | remote_receiver_ns = cg.esphome_ns.namespace('remote_receiver') | ||||||
| @@ -21,11 +22,15 @@ CONFIG_SCHEMA = remote_base.validate_triggers(cv.Schema({ | |||||||
|     cv.SplitDefault(CONF_BUFFER_SIZE, esp32='10000b', esp8266='1000b'): cv.validate_bytes, |     cv.SplitDefault(CONF_BUFFER_SIZE, esp32='10000b', esp8266='1000b'): cv.validate_bytes, | ||||||
|     cv.Optional(CONF_FILTER, default='50us'): cv.positive_time_period_microseconds, |     cv.Optional(CONF_FILTER, default='50us'): cv.positive_time_period_microseconds, | ||||||
|     cv.Optional(CONF_IDLE, default='10ms'): cv.positive_time_period_microseconds, |     cv.Optional(CONF_IDLE, default='10ms'): cv.positive_time_period_microseconds, | ||||||
|  |     cv.Optional(CONF_MEMORY_BLOCKS, default=3): cv.Range(min=1, max=8), | ||||||
| }).extend(cv.COMPONENT_SCHEMA)) | }).extend(cv.COMPONENT_SCHEMA)) | ||||||
|  |  | ||||||
|  |  | ||||||
| def to_code(config): | def to_code(config): | ||||||
|     pin = yield cg.gpio_pin_expression(config[CONF_PIN]) |     pin = yield cg.gpio_pin_expression(config[CONF_PIN]) | ||||||
|  |     if CORE.is_esp32: | ||||||
|  |         var = cg.new_Pvariable(config[CONF_ID], pin, config[CONF_MEMORY_BLOCKS]) | ||||||
|  |     else: | ||||||
|         var = cg.new_Pvariable(config[CONF_ID], pin) |         var = cg.new_Pvariable(config[CONF_ID], pin) | ||||||
|  |  | ||||||
|     yield remote_base.build_dumpers(config[CONF_DUMP]) |     yield remote_base.build_dumpers(config[CONF_DUMP]) | ||||||
|   | |||||||
| @@ -25,9 +25,20 @@ struct RemoteReceiverComponentStore { | |||||||
| }; | }; | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| class RemoteReceiverComponent : public remote_base::RemoteReceiverBase, public Component { | class RemoteReceiverComponent : public remote_base::RemoteReceiverBase, | ||||||
|  |                                 public Component | ||||||
|  | #ifdef ARDUINO_ARCH_ESP32 | ||||||
|  |     , | ||||||
|  |                                 public remote_base::RemoteRMTChannel | ||||||
|  | #endif | ||||||
|  | { | ||||||
|  public: |  public: | ||||||
|  | #ifdef ARDUINO_ARCH_ESP32 | ||||||
|  |   RemoteReceiverComponent(GPIOPin *pin, uint8_t mem_block_num = 1) | ||||||
|  |       : RemoteReceiverBase(pin), remote_base::RemoteRMTChannel(mem_block_num) {} | ||||||
|  | #else | ||||||
|   RemoteReceiverComponent(GPIOPin *pin) : RemoteReceiverBase(pin) {} |   RemoteReceiverComponent(GPIOPin *pin) : RemoteReceiverBase(pin) {} | ||||||
|  | #endif | ||||||
|   void setup() override; |   void setup() override; | ||||||
|   void dump_config() override; |   void dump_config() override; | ||||||
|   void loop() override; |   void loop() override; | ||||||
| @@ -40,11 +51,10 @@ class RemoteReceiverComponent : public remote_base::RemoteReceiverBase, public C | |||||||
|  protected: |  protected: | ||||||
| #ifdef ARDUINO_ARCH_ESP32 | #ifdef ARDUINO_ARCH_ESP32 | ||||||
|   void decode_rmt_(rmt_item32_t *item, size_t len); |   void decode_rmt_(rmt_item32_t *item, size_t len); | ||||||
|  |   RingbufHandle_t ringbuf_; | ||||||
|  |   esp_err_t error_code_{ESP_OK}; | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| #ifdef ARDUINO_ARCH_ESP32 |  | ||||||
|   RingbufHandle_t ringbuf_; |  | ||||||
| #endif |  | ||||||
| #ifdef ARDUINO_ARCH_ESP8266 | #ifdef ARDUINO_ARCH_ESP8266 | ||||||
|   RemoteReceiverComponentStore store_; |   RemoteReceiverComponentStore store_; | ||||||
|   HighFrequencyLoopRequester high_freq_; |   HighFrequencyLoopRequester high_freq_; | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ | |||||||
| #include "esphome/core/log.h" | #include "esphome/core/log.h" | ||||||
|  |  | ||||||
| #ifdef ARDUINO_ARCH_ESP32 | #ifdef ARDUINO_ARCH_ESP32 | ||||||
|  | #include <driver/rmt.h> | ||||||
|  |  | ||||||
| namespace esphome { | namespace esphome { | ||||||
| namespace remote_receiver { | namespace remote_receiver { | ||||||
| @@ -11,10 +12,8 @@ static const char *TAG = "remote_receiver.esp32"; | |||||||
| void RemoteReceiverComponent::setup() { | void RemoteReceiverComponent::setup() { | ||||||
|   ESP_LOGCONFIG(TAG, "Setting up Remote Receiver..."); |   ESP_LOGCONFIG(TAG, "Setting up Remote Receiver..."); | ||||||
|   rmt_config_t rmt{}; |   rmt_config_t rmt{}; | ||||||
|   rmt.channel = this->channel_; |   this->config_rmt(rmt); | ||||||
|   rmt.gpio_num = gpio_num_t(this->pin_->get_pin()); |   rmt.gpio_num = gpio_num_t(this->pin_->get_pin()); | ||||||
|   rmt.clk_div = this->clock_divider_; |  | ||||||
|   rmt.mem_block_num = 1; |  | ||||||
|   rmt.rmt_mode = RMT_MODE_RX; |   rmt.rmt_mode = RMT_MODE_RX; | ||||||
|   if (this->filter_us_ == 0) { |   if (this->filter_us_ == 0) { | ||||||
|     rmt.rx_config.filter_en = false; |     rmt.rx_config.filter_en = false; | ||||||
| @@ -58,6 +57,7 @@ void RemoteReceiverComponent::dump_config() { | |||||||
|                   "invert the signal using 'inverted: True' in the pin schema!"); |                   "invert the signal using 'inverted: True' in the pin schema!"); | ||||||
|   } |   } | ||||||
|   ESP_LOGCONFIG(TAG, "  Channel: %d", this->channel_); |   ESP_LOGCONFIG(TAG, "  Channel: %d", this->channel_); | ||||||
|  |   ESP_LOGCONFIG(TAG, "  RMT memory blocks: %d", this->mem_block_num_); | ||||||
|   ESP_LOGCONFIG(TAG, "  Clock divider: %u", this->clock_divider_); |   ESP_LOGCONFIG(TAG, "  Clock divider: %u", this->clock_divider_); | ||||||
|   ESP_LOGCONFIG(TAG, "  Tolerance: %u%%", this->tolerance_); |   ESP_LOGCONFIG(TAG, "  Tolerance: %u%%", this->tolerance_); | ||||||
|   ESP_LOGCONFIG(TAG, "  Filter out pulses shorter than: %u us", this->filter_us_); |   ESP_LOGCONFIG(TAG, "  Filter out pulses shorter than: %u us", this->filter_us_); | ||||||
|   | |||||||
| @@ -6,7 +6,13 @@ | |||||||
| namespace esphome { | namespace esphome { | ||||||
| namespace remote_transmitter { | namespace remote_transmitter { | ||||||
|  |  | ||||||
| class RemoteTransmitterComponent : public remote_base::RemoteTransmitterBase, public Component { | class RemoteTransmitterComponent : public remote_base::RemoteTransmitterBase, | ||||||
|  |                                    public Component | ||||||
|  | #ifdef ARDUINO_ARCH_ESP32 | ||||||
|  |     , | ||||||
|  |                                    public remote_base::RemoteRMTChannel | ||||||
|  | #endif | ||||||
|  | { | ||||||
|  public: |  public: | ||||||
|   explicit RemoteTransmitterComponent(GPIOPin *pin) : remote_base::RemoteTransmitterBase(pin) {} |   explicit RemoteTransmitterComponent(GPIOPin *pin) : remote_base::RemoteTransmitterBase(pin) {} | ||||||
|  |  | ||||||
| @@ -34,6 +40,7 @@ class RemoteTransmitterComponent : public remote_base::RemoteTransmitterBase, pu | |||||||
|   uint32_t current_carrier_frequency_{UINT32_MAX}; |   uint32_t current_carrier_frequency_{UINT32_MAX}; | ||||||
|   bool initialized_{false}; |   bool initialized_{false}; | ||||||
|   std::vector<rmt_item32_t> rmt_temp_; |   std::vector<rmt_item32_t> rmt_temp_; | ||||||
|  |   esp_err_t error_code_{ESP_OK}; | ||||||
| #endif | #endif | ||||||
|   uint8_t carrier_duty_percent_{50}; |   uint8_t carrier_duty_percent_{50}; | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -14,6 +14,7 @@ void RemoteTransmitterComponent::setup() {} | |||||||
| void RemoteTransmitterComponent::dump_config() { | void RemoteTransmitterComponent::dump_config() { | ||||||
|   ESP_LOGCONFIG(TAG, "Remote Transmitter..."); |   ESP_LOGCONFIG(TAG, "Remote Transmitter..."); | ||||||
|   ESP_LOGCONFIG(TAG, "  Channel: %d", this->channel_); |   ESP_LOGCONFIG(TAG, "  Channel: %d", this->channel_); | ||||||
|  |   ESP_LOGCONFIG(TAG, "  RMT memory blocks: %d", this->mem_block_num_); | ||||||
|   ESP_LOGCONFIG(TAG, "  Clock divider: %u", this->clock_divider_); |   ESP_LOGCONFIG(TAG, "  Clock divider: %u", this->clock_divider_); | ||||||
|   LOG_PIN("  Pin: ", this->pin_); |   LOG_PIN("  Pin: ", this->pin_); | ||||||
|  |  | ||||||
| @@ -29,11 +30,9 @@ void RemoteTransmitterComponent::dump_config() { | |||||||
| void RemoteTransmitterComponent::configure_rmt() { | void RemoteTransmitterComponent::configure_rmt() { | ||||||
|   rmt_config_t c{}; |   rmt_config_t c{}; | ||||||
|  |  | ||||||
|  |   this->config_rmt(c); | ||||||
|   c.rmt_mode = RMT_MODE_TX; |   c.rmt_mode = RMT_MODE_TX; | ||||||
|   c.channel = this->channel_; |  | ||||||
|   c.clk_div = this->clock_divider_; |  | ||||||
|   c.gpio_num = gpio_num_t(this->pin_->get_pin()); |   c.gpio_num = gpio_num_t(this->pin_->get_pin()); | ||||||
|   c.mem_block_num = 1; |  | ||||||
|   c.tx_config.loop_en = false; |   c.tx_config.loop_en = false; | ||||||
|  |  | ||||||
|   if (this->current_carrier_frequency_ == 0 || this->carrier_duty_percent_ == 100) { |   if (this->current_carrier_frequency_ == 0 || this->carrier_duty_percent_ == 100) { | ||||||
|   | |||||||
| @@ -250,6 +250,7 @@ CONF_MAX_VOLTAGE = 'max_voltage' | |||||||
| CONF_MEASUREMENT_DURATION = 'measurement_duration' | CONF_MEASUREMENT_DURATION = 'measurement_duration' | ||||||
| CONF_MEASUREMENT_SEQUENCE_NUMBER = 'measurement_sequence_number' | CONF_MEASUREMENT_SEQUENCE_NUMBER = 'measurement_sequence_number' | ||||||
| CONF_MEDIUM = 'medium' | CONF_MEDIUM = 'medium' | ||||||
|  | CONF_MEMORY_BLOCKS = 'memory_blocks' | ||||||
| CONF_METHOD = 'method' | CONF_METHOD = 'method' | ||||||
| CONF_MIN_LENGTH = 'min_length' | CONF_MIN_LENGTH = 'min_length' | ||||||
| CONF_MIN_LEVEL = 'min_level' | CONF_MIN_LEVEL = 'min_level' | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user