esp32 remote: make RMT memory blocks configureable (#1002)

This commit is contained in:
puuu 2020-05-01 10:59:51 +09:00 committed by GitHub
parent af66753c1b
commit d49ee47018
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 63 additions and 28 deletions

View file

@ -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) {

View file

@ -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:

View file

@ -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,12 +22,16 @@ 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])
var = cg.new_Pvariable(config[CONF_ID], 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)
yield remote_base.build_dumpers(config[CONF_DUMP]) yield remote_base.build_dumpers(config[CONF_DUMP])
yield remote_base.build_triggers(config) yield remote_base.build_triggers(config)

View file

@ -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_;

View file

@ -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_);

View file

@ -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};
}; };

View file

@ -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) {

View file

@ -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'