mirror of
https://github.com/esphome/esphome.git
synced 2025-02-21 20:38:16 +00:00
* Socket refactor and SSL * esp-idf temp * Fixes * Echo component and noise * Add noise API transport support * Updates * ESP-IDF * Complete * Fixes * Fixes * Versions update * New i2c APIs * Complete i2c refactor * SPI migration * Revert ESP Preferences migration, too complex for now * OTA support * Remove echo again * Remove ssl again * GPIOFlags updates * Rename esphal and ICACHE_RAM_ATTR * Make ESP32 arduino compilable again * Fix GPIO flags * Complete pin registry refactor and fixes * Fixes to make test1 compile * Remove sdkconfig file * Ignore sdkconfig file * Fixes in reviewing * Make test2 compile * Make test4 compile * Make test5 compile * Run clang-format * Fix lint errors * Use esp-idf APIs instead of btStart * Another round of fixes * Start implementing ESP8266 * Make test3 compile * Guard esp8266 code * Lint * Reformat * Fixes * Fixes v2 * more fixes * ESP-IDF tidy target * Convert ARDUINO_ARCH_ESPxx * Update WiFiSignalSensor * Update time ifdefs * OTA needs millis from hal * RestartSwitch needs delay from hal * ESP-IDF Uart * Fix OTA blank password * Allow setting sdkconfig * Fix idf partitions and allow setting sdkconfig from yaml * Re-add read/write compat APIs and fix esp8266 uart * Fix esp8266 store log strings in flash * Fix ESP32 arduino preferences not initialized * Update ifdefs * Change how sdkconfig change is detected * Add checks to ci-custom and fix them * Run clang-format * Add esp-idf clang-tidy target and fix errors * Fixes from clang-tidy idf round 2 * Fixes from compiling tests with esp-idf * Run clang-format * Switch test5.yaml to esp-idf * Implement ESP8266 Preferences * Lint * Re-do PIO package version selection a bit * Fix arduinoespressif32 package version * Fix unit tests * Lint * Lint fixes * Fix readv/writev not defined * Fix graphing component * Re-add all old options from core/config.py Co-authored-by: Jesse Hills <3060199+jesserockz@users.noreply.github.com>
202 lines
5.7 KiB
C++
202 lines
5.7 KiB
C++
#ifdef USE_ESP_IDF
|
|
|
|
#include "uart_component_esp_idf.h"
|
|
#include "esphome/core/application.h"
|
|
#include "esphome/core/defines.h"
|
|
#include "esphome/core/helpers.h"
|
|
#include "esphome/core/log.h"
|
|
|
|
#ifdef USE_LOGGER
|
|
#include "esphome/components/logger/logger.h"
|
|
#endif
|
|
|
|
namespace esphome {
|
|
namespace uart {
|
|
static const char *const TAG = "uart.idf";
|
|
|
|
uart_config_t IDFUARTComponent::get_config() {
|
|
uart_parity_t parity = UART_PARITY_DISABLE;
|
|
if (this->parity_ == UART_CONFIG_PARITY_EVEN)
|
|
parity = UART_PARITY_EVEN;
|
|
else if (this->parity_ == UART_CONFIG_PARITY_ODD)
|
|
parity = UART_PARITY_ODD;
|
|
|
|
uart_word_length_t data_bits;
|
|
switch (this->data_bits_) {
|
|
case 5:
|
|
data_bits = UART_DATA_5_BITS;
|
|
break;
|
|
case 6:
|
|
data_bits = UART_DATA_6_BITS;
|
|
break;
|
|
case 7:
|
|
data_bits = UART_DATA_7_BITS;
|
|
break;
|
|
case 8:
|
|
data_bits = UART_DATA_8_BITS;
|
|
break;
|
|
default:
|
|
data_bits = UART_DATA_BITS_MAX;
|
|
break;
|
|
}
|
|
|
|
uart_config_t uart_config;
|
|
uart_config.baud_rate = this->baud_rate_;
|
|
uart_config.data_bits = data_bits;
|
|
uart_config.parity = parity;
|
|
uart_config.stop_bits = this->stop_bits_ == 1 ? UART_STOP_BITS_1 : UART_STOP_BITS_2;
|
|
uart_config.flow_ctrl = UART_HW_FLOWCTRL_DISABLE;
|
|
uart_config.source_clk = UART_SCLK_APB;
|
|
uart_config.rx_flow_ctrl_thresh = 122;
|
|
|
|
return uart_config;
|
|
}
|
|
|
|
void IDFUARTComponent::setup() {
|
|
static uint8_t next_uart_num = 0;
|
|
#ifdef USE_LOGGER
|
|
if (logger::global_logger->get_uart_num() == next_uart_num)
|
|
next_uart_num++;
|
|
#endif
|
|
if (next_uart_num >= UART_NUM_MAX) {
|
|
ESP_LOGW(TAG, "Maximum number of UART components created already.");
|
|
this->mark_failed();
|
|
return;
|
|
}
|
|
this->uart_num_ = next_uart_num++;
|
|
ESP_LOGCONFIG(TAG, "Setting up UART %u...", this->uart_num_);
|
|
|
|
this->lock_ = xSemaphoreCreateMutex();
|
|
|
|
xSemaphoreTake(this->lock_, portMAX_DELAY);
|
|
|
|
uart_config_t uart_config = this->get_config();
|
|
esp_err_t err = uart_param_config(this->uart_num_, &uart_config);
|
|
if (err != ESP_OK) {
|
|
ESP_LOGW(TAG, "uart_param_config failed: %s", esp_err_to_name(err));
|
|
this->mark_failed();
|
|
return;
|
|
}
|
|
|
|
err = uart_driver_install(this->uart_num_, this->rx_buffer_size_, 0, 0, nullptr, 0);
|
|
if (err != ESP_OK) {
|
|
ESP_LOGW(TAG, "uart_driver_install failed: %s", esp_err_to_name(err));
|
|
this->mark_failed();
|
|
return;
|
|
}
|
|
|
|
int8_t tx = this->tx_pin_ != nullptr ? this->tx_pin_->get_pin() : -1;
|
|
int8_t rx = this->rx_pin_ != nullptr ? this->rx_pin_->get_pin() : -1;
|
|
|
|
err = uart_set_pin(this->uart_num_, tx, rx, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE);
|
|
if (err != ESP_OK) {
|
|
ESP_LOGW(TAG, "uart_set_pin failed: %s", esp_err_to_name(err));
|
|
this->mark_failed();
|
|
return;
|
|
}
|
|
|
|
uint32_t invert = 0;
|
|
if (this->tx_pin_ != nullptr && this->tx_pin_->is_inverted())
|
|
invert |= UART_SIGNAL_TXD_INV;
|
|
if (this->rx_pin_ != nullptr && this->rx_pin_->is_inverted())
|
|
invert |= UART_SIGNAL_RXD_INV;
|
|
|
|
err = uart_set_line_inverse(this->uart_num_, invert);
|
|
if (err != ESP_OK) {
|
|
ESP_LOGW(TAG, "uart_set_line_inverse failed: %s", esp_err_to_name(err));
|
|
this->mark_failed();
|
|
return;
|
|
}
|
|
|
|
xSemaphoreGive(this->lock_);
|
|
}
|
|
|
|
void IDFUARTComponent::dump_config() {
|
|
ESP_LOGCONFIG(TAG, "UART Bus:");
|
|
ESP_LOGCONFIG(TAG, " Number: %u", this->uart_num_);
|
|
LOG_PIN(" TX Pin: ", tx_pin_);
|
|
LOG_PIN(" RX Pin: ", rx_pin_);
|
|
if (this->rx_pin_ != nullptr) {
|
|
ESP_LOGCONFIG(TAG, " RX Buffer Size: %u", this->rx_buffer_size_);
|
|
}
|
|
ESP_LOGCONFIG(TAG, " Baud Rate: %u baud", this->baud_rate_);
|
|
ESP_LOGCONFIG(TAG, " Data Bits: %u", this->data_bits_);
|
|
ESP_LOGCONFIG(TAG, " Parity: %s", LOG_STR_ARG(parity_to_str(this->parity_)));
|
|
ESP_LOGCONFIG(TAG, " Stop bits: %u", this->stop_bits_);
|
|
this->check_logger_conflict();
|
|
}
|
|
|
|
void IDFUARTComponent::write_array(const uint8_t *data, size_t len) {
|
|
xSemaphoreTake(this->lock_, portMAX_DELAY);
|
|
uart_write_bytes(this->uart_num_, data, len);
|
|
xSemaphoreGive(this->lock_);
|
|
for (size_t i = 0; i < len; i++) {
|
|
ESP_LOGVV(TAG, " Wrote 0b" BYTE_TO_BINARY_PATTERN " (0x%02X)", BYTE_TO_BINARY(data[i]), data[i]);
|
|
}
|
|
}
|
|
bool IDFUARTComponent::peek_byte(uint8_t *data) {
|
|
if (!this->check_read_timeout_())
|
|
return false;
|
|
xSemaphoreTake(this->lock_, portMAX_DELAY);
|
|
if (this->has_peek_)
|
|
*data = this->peek_byte_;
|
|
else {
|
|
int len = uart_read_bytes(this->uart_num_, data, 1, 20 / portTICK_RATE_MS);
|
|
if (len == 0) {
|
|
*data = 0;
|
|
} else {
|
|
this->has_peek_ = true;
|
|
this->peek_byte_ = *data;
|
|
}
|
|
}
|
|
xSemaphoreGive(this->lock_);
|
|
return true;
|
|
}
|
|
bool IDFUARTComponent::read_array(uint8_t *data, size_t len) {
|
|
size_t length_to_read = len;
|
|
if (!this->check_read_timeout_(len))
|
|
return false;
|
|
xSemaphoreTake(this->lock_, portMAX_DELAY);
|
|
if (this->has_peek_) {
|
|
length_to_read--;
|
|
*data = this->peek_byte_;
|
|
data++;
|
|
this->has_peek_ = false;
|
|
}
|
|
if (length_to_read > 0)
|
|
uart_read_bytes(this->uart_num_, data, length_to_read, 20 / portTICK_RATE_MS);
|
|
|
|
xSemaphoreGive(this->lock_);
|
|
for (size_t i = 0; i < len; i++) {
|
|
ESP_LOGVV(TAG, " Read 0b" BYTE_TO_BINARY_PATTERN " (0x%02X)", BYTE_TO_BINARY(data[i]), data[i]);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
int IDFUARTComponent::available() {
|
|
size_t available;
|
|
|
|
xSemaphoreTake(this->lock_, portMAX_DELAY);
|
|
uart_get_buffered_data_len(this->uart_num_, &available);
|
|
if (this->has_peek_)
|
|
available++;
|
|
xSemaphoreGive(this->lock_);
|
|
|
|
return available;
|
|
}
|
|
|
|
void IDFUARTComponent::flush() {
|
|
ESP_LOGVV(TAG, " Flushing...");
|
|
xSemaphoreTake(this->lock_, portMAX_DELAY);
|
|
uart_wait_tx_done(this->uart_num_, portMAX_DELAY);
|
|
xSemaphoreGive(this->lock_);
|
|
}
|
|
|
|
void IDFUARTComponent::check_logger_conflict() {}
|
|
|
|
} // namespace uart
|
|
} // namespace esphome
|
|
|
|
#endif // USE_ESP32
|