mirror of
https://github.com/esphome/esphome.git
synced 2024-11-25 08:28:12 +01:00
esp32 remote: make RMT memory blocks configureable (#1002)
This commit is contained in:
parent
af66753c1b
commit
d49ee47018
8 changed files with 63 additions and 28 deletions
|
@ -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,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)
|
||||||
|
|
|
@ -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'
|
||||||
|
|
Loading…
Reference in a new issue