mirror of
				https://github.com/esphome/esphome.git
				synced 2025-11-04 00:51:49 +00:00 
			
		
		
		
	Compare commits
	
		
			15 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					0104bf3fc8 | ||
| 
						 | 
					9b1e1bf56c | ||
| 
						 | 
					33e0f16b3b | ||
| 
						 | 
					0807d60c6a | ||
| 
						 | 
					f018fde369 | ||
| 
						 | 
					c47f8fc02c | ||
| 
						 | 
					76ab923780 | ||
| 
						 | 
					11dba3147d | ||
| 
						 | 
					8c2d9101d5 | ||
| 
						 | 
					61b8004536 | ||
| 
						 | 
					db02c4ea21 | ||
| 
						 | 
					f077a5962d | ||
| 
						 | 
					fa4ba43eb9 | ||
| 
						 | 
					9579423b24 | ||
| 
						 | 
					02449f24c9 | 
@@ -225,7 +225,7 @@ esphome/components/pn532_spi/* @OttoWinter @jesserockz
 | 
			
		||||
esphome/components/power_supply/* @esphome/core
 | 
			
		||||
esphome/components/preferences/* @esphome/core
 | 
			
		||||
esphome/components/psram/* @esphome/core
 | 
			
		||||
esphome/components/pulse_meter/* @cstaahl @stevebaxter
 | 
			
		||||
esphome/components/pulse_meter/* @TrentHouliston @cstaahl @stevebaxter
 | 
			
		||||
esphome/components/pvvx_mithermometer/* @pasiz
 | 
			
		||||
esphome/components/qmp6988/* @andrewpc
 | 
			
		||||
esphome/components/qr_code/* @wjtje
 | 
			
		||||
 
 | 
			
		||||
@@ -17,11 +17,12 @@ CONF_ON_FRAME = "on_frame"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def validate_id(config):
 | 
			
		||||
    can_id = config[CONF_CAN_ID]
 | 
			
		||||
    id_ext = config[CONF_USE_EXTENDED_ID]
 | 
			
		||||
    if not id_ext:
 | 
			
		||||
        if can_id > 0x7FF:
 | 
			
		||||
            raise cv.Invalid("Standard IDs must be 11 Bit (0x000-0x7ff / 0-2047)")
 | 
			
		||||
    if CONF_CAN_ID in config:
 | 
			
		||||
        can_id = config[CONF_CAN_ID]
 | 
			
		||||
        id_ext = config[CONF_USE_EXTENDED_ID]
 | 
			
		||||
        if not id_ext:
 | 
			
		||||
            if can_id > 0x7FF:
 | 
			
		||||
                raise cv.Invalid("Standard IDs must be 11 Bit (0x000-0x7ff / 0-2047)")
 | 
			
		||||
    return config
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -151,22 +152,18 @@ async def canbus_action_to_code(config, action_id, template_arg, args):
 | 
			
		||||
    if can_id := config.get(CONF_CAN_ID):
 | 
			
		||||
        can_id = await cg.templatable(can_id, args, cg.uint32)
 | 
			
		||||
        cg.add(var.set_can_id(can_id))
 | 
			
		||||
    use_extended_id = await cg.templatable(
 | 
			
		||||
        config[CONF_USE_EXTENDED_ID], args, cg.uint32
 | 
			
		||||
    )
 | 
			
		||||
    cg.add(var.set_use_extended_id(use_extended_id))
 | 
			
		||||
        cg.add(var.set_use_extended_id(config[CONF_USE_EXTENDED_ID]))
 | 
			
		||||
 | 
			
		||||
    remote_transmission_request = await cg.templatable(
 | 
			
		||||
        config[CONF_REMOTE_TRANSMISSION_REQUEST], args, bool
 | 
			
		||||
    cg.add(
 | 
			
		||||
        var.set_remote_transmission_request(config[CONF_REMOTE_TRANSMISSION_REQUEST])
 | 
			
		||||
    )
 | 
			
		||||
    cg.add(var.set_remote_transmission_request(remote_transmission_request))
 | 
			
		||||
 | 
			
		||||
    data = config[CONF_DATA]
 | 
			
		||||
    if isinstance(data, bytes):
 | 
			
		||||
        data = [int(x) for x in data]
 | 
			
		||||
    if cg.is_template(data):
 | 
			
		||||
        templ = await cg.templatable(data, args, cg.std_vector.template(cg.uint8))
 | 
			
		||||
        cg.add(var.set_data_template(templ))
 | 
			
		||||
    else:
 | 
			
		||||
        if isinstance(data, bytes):
 | 
			
		||||
            data = [int(x) for x in data]
 | 
			
		||||
        cg.add(var.set_data_static(data))
 | 
			
		||||
    return var
 | 
			
		||||
 
 | 
			
		||||
@@ -213,6 +213,8 @@ ClimateCall &ClimateCall::set_preset(const std::string &preset) {
 | 
			
		||||
    this->set_preset(CLIMATE_PRESET_SLEEP);
 | 
			
		||||
  } else if (str_equals_case_insensitive(preset, "ACTIVITY")) {
 | 
			
		||||
    this->set_preset(CLIMATE_PRESET_ACTIVITY);
 | 
			
		||||
  } else if (str_equals_case_insensitive(preset, "NONE")) {
 | 
			
		||||
    this->set_preset(CLIMATE_PRESET_NONE);
 | 
			
		||||
  } else {
 | 
			
		||||
    if (this->parent_->get_traits().supports_custom_preset(preset)) {
 | 
			
		||||
      this->custom_preset_ = preset;
 | 
			
		||||
 
 | 
			
		||||
@@ -18,7 +18,7 @@ _ESP_SDIO_PINS = {
 | 
			
		||||
    11: "Flash Command",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
_ESP32_STRAPPING_PINS = {0, 2, 4, 12, 15}
 | 
			
		||||
_ESP32_STRAPPING_PINS = {0, 2, 5, 12, 15}
 | 
			
		||||
_LOGGER = logging.getLogger(__name__)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -36,6 +36,9 @@ CONFIG_SCHEMA = cv.Schema(
 | 
			
		||||
        cv.Optional(
 | 
			
		||||
            CONF_AUTHORIZED_DURATION, default="1min"
 | 
			
		||||
        ): cv.positive_time_period_milliseconds,
 | 
			
		||||
        cv.Optional(
 | 
			
		||||
            CONF_WIFI_TIMEOUT, default="1min"
 | 
			
		||||
        ): cv.positive_time_period_milliseconds,
 | 
			
		||||
    }
 | 
			
		||||
).extend(cv.COMPONENT_SCHEMA)
 | 
			
		||||
 | 
			
		||||
@@ -53,6 +56,8 @@ async def to_code(config):
 | 
			
		||||
    cg.add(var.set_identify_duration(config[CONF_IDENTIFY_DURATION]))
 | 
			
		||||
    cg.add(var.set_authorized_duration(config[CONF_AUTHORIZED_DURATION]))
 | 
			
		||||
 | 
			
		||||
    cg.add(var.set_wifi_timeout(config[CONF_WIFI_TIMEOUT]))
 | 
			
		||||
 | 
			
		||||
    if CONF_AUTHORIZER in config and config[CONF_AUTHORIZER] is not None:
 | 
			
		||||
        activator = await cg.get_variable(config[CONF_AUTHORIZER])
 | 
			
		||||
        cg.add(var.set_authorizer(activator))
 | 
			
		||||
 
 | 
			
		||||
@@ -51,6 +51,9 @@ class ESP32ImprovComponent : public Component, public BLEServiceComponent {
 | 
			
		||||
  void set_identify_duration(uint32_t identify_duration) { this->identify_duration_ = identify_duration; }
 | 
			
		||||
  void set_authorized_duration(uint32_t authorized_duration) { this->authorized_duration_ = authorized_duration; }
 | 
			
		||||
 | 
			
		||||
  void set_wifi_timeout(uint32_t wifi_timeout) { this->wifi_timeout_ = wifi_timeout; }
 | 
			
		||||
  uint32_t get_wifi_timeout() const { return this->wifi_timeout_; }
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  bool should_start_{false};
 | 
			
		||||
  bool setup_complete_{false};
 | 
			
		||||
@@ -60,6 +63,8 @@ class ESP32ImprovComponent : public Component, public BLEServiceComponent {
 | 
			
		||||
  uint32_t authorized_start_{0};
 | 
			
		||||
  uint32_t authorized_duration_;
 | 
			
		||||
 | 
			
		||||
  uint32_t wifi_timeout_{};
 | 
			
		||||
 | 
			
		||||
  std::vector<uint8_t> incoming_data_;
 | 
			
		||||
  wifi::WiFiAP connecting_sta_;
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,10 +1,14 @@
 | 
			
		||||
from esphome.components.logger import USB_CDC, USB_SERIAL_JTAG
 | 
			
		||||
from esphome.components import improv_base
 | 
			
		||||
from esphome.components.esp32 import get_esp32_variant
 | 
			
		||||
from esphome.components.esp32.const import (
 | 
			
		||||
    VARIANT_ESP32S3,
 | 
			
		||||
)
 | 
			
		||||
from esphome.components.logger import USB_CDC
 | 
			
		||||
from esphome.const import CONF_BAUD_RATE, CONF_HARDWARE_UART, CONF_ID, CONF_LOGGER
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.core import CORE
 | 
			
		||||
import esphome.final_validate as fv
 | 
			
		||||
from esphome.components import improv_base
 | 
			
		||||
 | 
			
		||||
AUTO_LOAD = ["improv_base"]
 | 
			
		||||
CODEOWNERS = ["@esphome/core"]
 | 
			
		||||
@@ -30,7 +34,10 @@ def validate_logger(config):
 | 
			
		||||
    if logger_conf[CONF_BAUD_RATE] == 0:
 | 
			
		||||
        raise cv.Invalid("improv_serial requires the logger baud_rate to be not 0")
 | 
			
		||||
    if CORE.using_esp_idf:
 | 
			
		||||
        if logger_conf[CONF_HARDWARE_UART] in [USB_SERIAL_JTAG, USB_CDC]:
 | 
			
		||||
        if (
 | 
			
		||||
            logger_conf[CONF_HARDWARE_UART] == USB_CDC
 | 
			
		||||
            and get_esp32_variant() == VARIANT_ESP32S3
 | 
			
		||||
        ):
 | 
			
		||||
            raise cv.Invalid(
 | 
			
		||||
                "improv_serial does not support the selected logger hardware_uart"
 | 
			
		||||
            )
 | 
			
		||||
 
 | 
			
		||||
@@ -31,26 +31,57 @@ void ImprovSerialComponent::setup() {
 | 
			
		||||
 | 
			
		||||
void ImprovSerialComponent::dump_config() { ESP_LOGCONFIG(TAG, "Improv Serial:"); }
 | 
			
		||||
 | 
			
		||||
int ImprovSerialComponent::available_() {
 | 
			
		||||
optional<uint8_t> ImprovSerialComponent::read_byte_() {
 | 
			
		||||
  optional<uint8_t> byte;
 | 
			
		||||
  uint8_t data = 0;
 | 
			
		||||
#ifdef USE_ARDUINO
 | 
			
		||||
  return this->hw_serial_->available();
 | 
			
		||||
  if (this->hw_serial_->available()) {
 | 
			
		||||
    this->hw_serial_->readBytes(&data, 1);
 | 
			
		||||
    byte = data;
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ESP_IDF
 | 
			
		||||
  size_t available;
 | 
			
		||||
  uart_get_buffered_data_len(this->uart_num_, &available);
 | 
			
		||||
  return available;
 | 
			
		||||
  switch (logger::global_logger->get_uart()) {
 | 
			
		||||
    case logger::UART_SELECTION_UART0:
 | 
			
		||||
    case logger::UART_SELECTION_UART1:
 | 
			
		||||
#if !defined(USE_ESP32_VARIANT_ESP32C3) && !defined(USE_ESP32_VARIANT_ESP32C6) && \
 | 
			
		||||
    !defined(USE_ESP32_VARIANT_ESP32S2) && !defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
    case logger::UART_SELECTION_UART2:
 | 
			
		||||
#endif  // !USE_ESP32_VARIANT_ESP32C3 && !USE_ESP32_VARIANT_ESP32S2 && !USE_ESP32_VARIANT_ESP32S3
 | 
			
		||||
      if (this->uart_num_ >= 0) {
 | 
			
		||||
        size_t available;
 | 
			
		||||
        uart_get_buffered_data_len(this->uart_num_, &available);
 | 
			
		||||
        if (available) {
 | 
			
		||||
          uart_read_bytes(this->uart_num_, &data, 1, 20 / portTICK_PERIOD_MS);
 | 
			
		||||
          byte = data;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      break;
 | 
			
		||||
#if defined(CONFIG_ESP_CONSOLE_USB_CDC) && (defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3))
 | 
			
		||||
    case logger::UART_SELECTION_USB_CDC:
 | 
			
		||||
#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 0, 0)
 | 
			
		||||
      if (esp_usb_console_available_for_read()) {
 | 
			
		||||
#else
 | 
			
		||||
      if (esp_usb_console_read_available()) {
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint8_t ImprovSerialComponent::read_byte_() {
 | 
			
		||||
  uint8_t data;
 | 
			
		||||
#ifdef USE_ARDUINO
 | 
			
		||||
  this->hw_serial_->readBytes(&data, 1);
 | 
			
		||||
        esp_usb_console_read_buf((char *) &data, 1);
 | 
			
		||||
        byte = data;
 | 
			
		||||
      }
 | 
			
		||||
      break;
 | 
			
		||||
#endif  // USE_ESP32_VARIANT_ESP32S2 || USE_ESP32_VARIANT_ESP32S3
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32C6) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
    case logger::UART_SELECTION_USB_SERIAL_JTAG: {
 | 
			
		||||
      if (usb_serial_jtag_read_bytes((char *) &data, 1, 20 / portTICK_PERIOD_MS)) {
 | 
			
		||||
        byte = data;
 | 
			
		||||
      }
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
#endif  // USE_ESP32_VARIANT_ESP32C3 || USE_ESP32_VARIANT_ESP32C6 || USE_ESP32_VARIANT_ESP32S3
 | 
			
		||||
    default:
 | 
			
		||||
      break;
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ESP_IDF
 | 
			
		||||
  uart_read_bytes(this->uart_num_, &data, 1, 20 / portTICK_PERIOD_MS);
 | 
			
		||||
#endif
 | 
			
		||||
  return data;
 | 
			
		||||
  return byte;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ImprovSerialComponent::write_data_(std::vector<uint8_t> &data) {
 | 
			
		||||
@@ -59,24 +90,49 @@ void ImprovSerialComponent::write_data_(std::vector<uint8_t> &data) {
 | 
			
		||||
  this->hw_serial_->write(data.data(), data.size());
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ESP_IDF
 | 
			
		||||
  uart_write_bytes(this->uart_num_, data.data(), data.size());
 | 
			
		||||
  switch (logger::global_logger->get_uart()) {
 | 
			
		||||
    case logger::UART_SELECTION_UART0:
 | 
			
		||||
    case logger::UART_SELECTION_UART1:
 | 
			
		||||
#if !defined(USE_ESP32_VARIANT_ESP32C3) && !defined(USE_ESP32_VARIANT_ESP32C6) && \
 | 
			
		||||
    !defined(USE_ESP32_VARIANT_ESP32S2) && !defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
    case logger::UART_SELECTION_UART2:
 | 
			
		||||
#endif  // !USE_ESP32_VARIANT_ESP32C3 && !USE_ESP32_VARIANT_ESP32S2 && !USE_ESP32_VARIANT_ESP32S3
 | 
			
		||||
      uart_write_bytes(this->uart_num_, data.data(), data.size());
 | 
			
		||||
      break;
 | 
			
		||||
#if defined(CONFIG_ESP_CONSOLE_USB_CDC) && (defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3))
 | 
			
		||||
    case logger::UART_SELECTION_USB_CDC: {
 | 
			
		||||
      const char *msg = (char *) data.data();
 | 
			
		||||
      esp_usb_console_write_buf(msg, data.size());
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
#endif  // USE_ESP32_VARIANT_ESP32S2 || USE_ESP32_VARIANT_ESP32S3
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32C6) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
    case logger::UART_SELECTION_USB_SERIAL_JTAG:
 | 
			
		||||
      usb_serial_jtag_write_bytes((char *) data.data(), data.size(), 20 / portTICK_PERIOD_MS);
 | 
			
		||||
      break;
 | 
			
		||||
#endif  // USE_ESP32_VARIANT_ESP32C3 || USE_ESP32_VARIANT_ESP32S3
 | 
			
		||||
    default:
 | 
			
		||||
      break;
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void ImprovSerialComponent::loop() {
 | 
			
		||||
  const uint32_t now = millis();
 | 
			
		||||
  if (now - this->last_read_byte_ > 50) {
 | 
			
		||||
  if (this->last_read_byte_ && (millis() - this->last_read_byte_ > IMPROV_SERIAL_TIMEOUT)) {
 | 
			
		||||
    this->last_read_byte_ = 0;
 | 
			
		||||
    this->rx_buffer_.clear();
 | 
			
		||||
    this->last_read_byte_ = now;
 | 
			
		||||
    ESP_LOGV(TAG, "Improv Serial timeout");
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  while (this->available_()) {
 | 
			
		||||
    uint8_t byte = this->read_byte_();
 | 
			
		||||
    if (this->parse_improv_serial_byte_(byte)) {
 | 
			
		||||
      this->last_read_byte_ = now;
 | 
			
		||||
  auto byte = this->read_byte_();
 | 
			
		||||
  while (byte.has_value()) {
 | 
			
		||||
    if (this->parse_improv_serial_byte_(byte.value())) {
 | 
			
		||||
      this->last_read_byte_ = millis();
 | 
			
		||||
    } else {
 | 
			
		||||
      this->last_read_byte_ = 0;
 | 
			
		||||
      this->rx_buffer_.clear();
 | 
			
		||||
    }
 | 
			
		||||
    byte = this->read_byte_();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (this->state_ == improv::STATE_PROVISIONING) {
 | 
			
		||||
 
 | 
			
		||||
@@ -14,6 +14,13 @@
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ESP_IDF
 | 
			
		||||
#include <driver/uart.h>
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32C6) || defined(USE_ESP32_VARIANT_ESP32S3) || \
 | 
			
		||||
    defined(USE_ESP32_VARIANT_ESP32H2)
 | 
			
		||||
#include <driver/usb_serial_jtag.h>
 | 
			
		||||
#endif
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
#include <esp_private/usb_console.h>
 | 
			
		||||
#endif
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
@@ -26,6 +33,7 @@ enum ImprovSerialType : uint8_t {
 | 
			
		||||
  TYPE_RPC_RESPONSE = 0x04
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static const uint16_t IMPROV_SERIAL_TIMEOUT = 100;
 | 
			
		||||
static const uint8_t IMPROV_SERIAL_VERSION = 1;
 | 
			
		||||
 | 
			
		||||
class ImprovSerialComponent : public Component, public improv_base::ImprovBase {
 | 
			
		||||
@@ -48,8 +56,7 @@ class ImprovSerialComponent : public Component, public improv_base::ImprovBase {
 | 
			
		||||
  std::vector<uint8_t> build_rpc_settings_response_(improv::Command command);
 | 
			
		||||
  std::vector<uint8_t> build_version_info_();
 | 
			
		||||
 | 
			
		||||
  int available_();
 | 
			
		||||
  uint8_t read_byte_();
 | 
			
		||||
  optional<uint8_t> read_byte_();
 | 
			
		||||
  void write_data_(std::vector<uint8_t> &data);
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ARDUINO
 | 
			
		||||
 
 | 
			
		||||
@@ -3,8 +3,21 @@
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP_IDF
 | 
			
		||||
#include <driver/uart.h>
 | 
			
		||||
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32C6) || defined(USE_ESP32_VARIANT_ESP32S3) || \
 | 
			
		||||
    defined(USE_ESP32_VARIANT_ESP32H2)
 | 
			
		||||
#include <driver/usb_serial_jtag.h>
 | 
			
		||||
#include <esp_vfs_dev.h>
 | 
			
		||||
#include <esp_vfs_usb_serial_jtag.h>
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#include "freertos/FreeRTOS.h"
 | 
			
		||||
#include "esp_idf_version.h"
 | 
			
		||||
 | 
			
		||||
#include <cstdint>
 | 
			
		||||
#include <cstdio>
 | 
			
		||||
#include <fcntl.h>
 | 
			
		||||
 | 
			
		||||
#endif  // USE_ESP_IDF
 | 
			
		||||
 | 
			
		||||
#if defined(USE_ESP32_FRAMEWORK_ARDUINO) || defined(USE_ESP_IDF)
 | 
			
		||||
@@ -93,6 +106,58 @@ void Logger::log_vprintf_(int level, const char *tag, int line, const __FlashStr
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP_IDF
 | 
			
		||||
void Logger::init_uart_() {
 | 
			
		||||
  uart_config_t uart_config{};
 | 
			
		||||
  uart_config.baud_rate = (int) baud_rate_;
 | 
			
		||||
  uart_config.data_bits = UART_DATA_8_BITS;
 | 
			
		||||
  uart_config.parity = UART_PARITY_DISABLE;
 | 
			
		||||
  uart_config.stop_bits = UART_STOP_BITS_1;
 | 
			
		||||
  uart_config.flow_ctrl = UART_HW_FLOWCTRL_DISABLE;
 | 
			
		||||
#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 0, 0)
 | 
			
		||||
  uart_config.source_clk = UART_SCLK_DEFAULT;
 | 
			
		||||
#endif
 | 
			
		||||
  uart_param_config(this->uart_num_, &uart_config);
 | 
			
		||||
  const int uart_buffer_size = tx_buffer_size_;
 | 
			
		||||
  // Install UART driver using an event queue here
 | 
			
		||||
  uart_driver_install(this->uart_num_, uart_buffer_size, uart_buffer_size, 10, nullptr, 0);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
void Logger::init_usb_cdc_() {}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32C6) || defined(USE_ESP32_VARIANT_ESP32S3) || \
 | 
			
		||||
    defined(USE_ESP32_VARIANT_ESP32H2)
 | 
			
		||||
void Logger::init_usb_serial_jtag_() {
 | 
			
		||||
  setvbuf(stdin, NULL, _IONBF, 0);  // Disable buffering on stdin
 | 
			
		||||
 | 
			
		||||
  // Minicom, screen, idf_monitor send CR when ENTER key is pressed
 | 
			
		||||
  esp_vfs_dev_usb_serial_jtag_set_rx_line_endings(ESP_LINE_ENDINGS_CR);
 | 
			
		||||
  // Move the caret to the beginning of the next line on '\n'
 | 
			
		||||
  esp_vfs_dev_usb_serial_jtag_set_tx_line_endings(ESP_LINE_ENDINGS_CRLF);
 | 
			
		||||
 | 
			
		||||
  // Enable non-blocking mode on stdin and stdout
 | 
			
		||||
  fcntl(fileno(stdout), F_SETFL, 0);
 | 
			
		||||
  fcntl(fileno(stdin), F_SETFL, 0);
 | 
			
		||||
 | 
			
		||||
  usb_serial_jtag_driver_config_t usb_serial_jtag_config{};
 | 
			
		||||
  usb_serial_jtag_config.rx_buffer_size = 512;
 | 
			
		||||
  usb_serial_jtag_config.tx_buffer_size = 512;
 | 
			
		||||
 | 
			
		||||
  esp_err_t ret = ESP_OK;
 | 
			
		||||
  // Install USB-SERIAL-JTAG driver for interrupt-driven reads and writes
 | 
			
		||||
  ret = usb_serial_jtag_driver_install(&usb_serial_jtag_config);
 | 
			
		||||
  if (ret != ESP_OK) {
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Tell vfs to use usb-serial-jtag driver
 | 
			
		||||
  esp_vfs_usb_serial_jtag_use_driver();
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
int HOT Logger::level_for(const char *tag) {
 | 
			
		||||
  // Uses std::vector<> for low memory footprint, though the vector
 | 
			
		||||
  // could be sorted to minimize lookup times. This feature isn't used that
 | 
			
		||||
@@ -120,19 +185,19 @@ void HOT Logger::log_message_(int level, const char *tag, int offset) {
 | 
			
		||||
#ifdef USE_ESP_IDF
 | 
			
		||||
    if (
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32S2)
 | 
			
		||||
        uart_ == UART_SELECTION_USB_CDC
 | 
			
		||||
        this->uart_ == UART_SELECTION_USB_CDC
 | 
			
		||||
#elif defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32C6) || defined(USE_ESP32_VARIANT_ESP32H2)
 | 
			
		||||
        uart_ == UART_SELECTION_USB_SERIAL_JTAG
 | 
			
		||||
        this->uart_ == UART_SELECTION_USB_SERIAL_JTAG
 | 
			
		||||
#elif defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
        uart_ == UART_SELECTION_USB_CDC || uart_ == UART_SELECTION_USB_SERIAL_JTAG
 | 
			
		||||
        this->uart_ == UART_SELECTION_USB_CDC || this->uart_ == UART_SELECTION_USB_SERIAL_JTAG
 | 
			
		||||
#else
 | 
			
		||||
        /* DISABLES CODE */ (false)  // NOLINT
 | 
			
		||||
#endif
 | 
			
		||||
    ) {
 | 
			
		||||
      puts(msg);
 | 
			
		||||
    } else {
 | 
			
		||||
      uart_write_bytes(uart_num_, msg, strlen(msg));
 | 
			
		||||
      uart_write_bytes(uart_num_, "\n", 1);
 | 
			
		||||
      uart_write_bytes(this->uart_num_, msg, strlen(msg));
 | 
			
		||||
      uart_write_bytes(this->uart_num_, "\n", 1);
 | 
			
		||||
    }
 | 
			
		||||
#endif
 | 
			
		||||
  }
 | 
			
		||||
@@ -209,48 +274,38 @@ void Logger::pre_setup() {
 | 
			
		||||
    }
 | 
			
		||||
#endif  // USE_ARDUINO
 | 
			
		||||
#ifdef USE_ESP_IDF
 | 
			
		||||
    uart_num_ = UART_NUM_0;
 | 
			
		||||
    switch (uart_) {
 | 
			
		||||
    this->uart_num_ = UART_NUM_0;
 | 
			
		||||
    switch (this->uart_) {
 | 
			
		||||
      case UART_SELECTION_UART0:
 | 
			
		||||
        uart_num_ = UART_NUM_0;
 | 
			
		||||
        this->uart_num_ = UART_NUM_0;
 | 
			
		||||
        break;
 | 
			
		||||
      case UART_SELECTION_UART1:
 | 
			
		||||
        uart_num_ = UART_NUM_1;
 | 
			
		||||
        this->uart_num_ = UART_NUM_1;
 | 
			
		||||
        break;
 | 
			
		||||
#if !defined(USE_ESP32_VARIANT_ESP32C3) && !defined(USE_ESP32_VARIANT_ESP32C6) && \
 | 
			
		||||
    !defined(USE_ESP32_VARIANT_ESP32S2) && !defined(USE_ESP32_VARIANT_ESP32S3) && !defined(USE_ESP32_VARIANT_ESP32H2)
 | 
			
		||||
      case UART_SELECTION_UART2:
 | 
			
		||||
        uart_num_ = UART_NUM_2;
 | 
			
		||||
        this->uart_num_ = UART_NUM_2;
 | 
			
		||||
        break;
 | 
			
		||||
#endif  // !USE_ESP32_VARIANT_ESP32C3 && !USE_ESP32_VARIANT_ESP32S2 && !USE_ESP32_VARIANT_ESP32S3 &&
 | 
			
		||||
        // !USE_ESP32_VARIANT_ESP32H2
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
      case UART_SELECTION_USB_CDC:
 | 
			
		||||
        uart_num_ = -1;
 | 
			
		||||
        this->uart_num_ = -1;
 | 
			
		||||
        this->init_usb_cdc_();
 | 
			
		||||
        break;
 | 
			
		||||
#endif  // USE_ESP32_VARIANT_ESP32S2 || USE_ESP32_VARIANT_ESP32S3
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32C6) || defined(USE_ESP32_VARIANT_ESP32S3) || \
 | 
			
		||||
    defined(USE_ESP32_VARIANT_ESP32H2)
 | 
			
		||||
      case UART_SELECTION_USB_SERIAL_JTAG:
 | 
			
		||||
        uart_num_ = -1;
 | 
			
		||||
        this->uart_num_ = -1;
 | 
			
		||||
        this->init_usb_serial_jtag_();
 | 
			
		||||
        break;
 | 
			
		||||
#endif  // USE_ESP32_VARIANT_ESP32C3 || USE_ESP32_VARIANT_ESP32C6 || USE_ESP32_VARIANT_ESP32S3 ||
 | 
			
		||||
        // USE_ESP32_VARIANT_ESP32H2
 | 
			
		||||
    }
 | 
			
		||||
    if (uart_num_ >= 0) {
 | 
			
		||||
      uart_config_t uart_config{};
 | 
			
		||||
      uart_config.baud_rate = (int) baud_rate_;
 | 
			
		||||
      uart_config.data_bits = UART_DATA_8_BITS;
 | 
			
		||||
      uart_config.parity = UART_PARITY_DISABLE;
 | 
			
		||||
      uart_config.stop_bits = UART_STOP_BITS_1;
 | 
			
		||||
      uart_config.flow_ctrl = UART_HW_FLOWCTRL_DISABLE;
 | 
			
		||||
#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 0, 0)
 | 
			
		||||
      uart_config.source_clk = UART_SCLK_DEFAULT;
 | 
			
		||||
#endif
 | 
			
		||||
      uart_param_config(uart_num_, &uart_config);
 | 
			
		||||
      const int uart_buffer_size = tx_buffer_size_;
 | 
			
		||||
      // Install UART driver using an event queue here
 | 
			
		||||
      uart_driver_install(uart_num_, uart_buffer_size, uart_buffer_size, 10, nullptr, 0);
 | 
			
		||||
    if (this->uart_num_ >= 0) {
 | 
			
		||||
      this->init_uart_();
 | 
			
		||||
    }
 | 
			
		||||
#endif  // USE_ESP_IDF
 | 
			
		||||
  }
 | 
			
		||||
 
 | 
			
		||||
@@ -107,6 +107,16 @@ class Logger : public Component {
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
#ifdef USE_ESP_IDF
 | 
			
		||||
  void init_uart_();
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3)
 | 
			
		||||
  void init_usb_cdc_();
 | 
			
		||||
#endif
 | 
			
		||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32C6) || defined(USE_ESP32_VARIANT_ESP32S3) || \
 | 
			
		||||
    defined(USE_ESP32_VARIANT_ESP32H2)
 | 
			
		||||
  void init_usb_serial_jtag_();
 | 
			
		||||
#endif
 | 
			
		||||
#endif
 | 
			
		||||
  void write_header_(int level, const char *tag, int line);
 | 
			
		||||
  void write_footer_();
 | 
			
		||||
  void log_message_(int level, const char *tag, int offset = 0);
 | 
			
		||||
 
 | 
			
		||||
@@ -19,9 +19,7 @@ class MQTTBackendLibreTiny final : public MQTTBackend {
 | 
			
		||||
  void set_will(const char *topic, uint8_t qos, bool retain, const char *payload) final {
 | 
			
		||||
    mqtt_client_.setWill(topic, qos, retain, payload);
 | 
			
		||||
  }
 | 
			
		||||
  void set_server(network::IPAddress ip, uint16_t port) final {
 | 
			
		||||
    mqtt_client_.setServer(IPAddress(static_cast<uint32_t>(ip)), port);
 | 
			
		||||
  }
 | 
			
		||||
  void set_server(network::IPAddress ip, uint16_t port) final { mqtt_client_.setServer(IPAddress(ip), port); }
 | 
			
		||||
  void set_server(const char *host, uint16_t port) final { mqtt_client_.setServer(host, port); }
 | 
			
		||||
#if ASYNC_TCP_SSL_ENABLED
 | 
			
		||||
  void set_secure(bool secure) { mqtt_client.setSecure(secure); }
 | 
			
		||||
 
 | 
			
		||||
@@ -34,7 +34,12 @@ struct IPAddress {
 | 
			
		||||
  }
 | 
			
		||||
  IPAddress(const ip_addr_t *other_ip) { ip_addr_copy(ip_addr_, *other_ip); }
 | 
			
		||||
  IPAddress(const std::string &in_address) { ipaddr_aton(in_address.c_str(), &ip_addr_); }
 | 
			
		||||
  IPAddress(ip4_addr_t *other_ip) { memcpy((void *) &ip_addr_, (void *) other_ip, sizeof(ip4_addr_t)); }
 | 
			
		||||
  IPAddress(ip4_addr_t *other_ip) {
 | 
			
		||||
    memcpy((void *) &ip_addr_, (void *) other_ip, sizeof(ip4_addr_t));
 | 
			
		||||
#if USE_ESP32
 | 
			
		||||
    ip_addr_.type = IPADDR_TYPE_V4;
 | 
			
		||||
#endif
 | 
			
		||||
  }
 | 
			
		||||
#if USE_ARDUINO
 | 
			
		||||
  IPAddress(const arduino_ns::IPAddress &other_ip) { ip_addr_set_ip4_u32(&ip_addr_, other_ip); }
 | 
			
		||||
#endif
 | 
			
		||||
 
 | 
			
		||||
@@ -7,6 +7,13 @@ namespace pulse_meter {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "pulse_meter";
 | 
			
		||||
 | 
			
		||||
void PulseMeterSensor::set_total_pulses(uint32_t pulses) {
 | 
			
		||||
  this->total_pulses_ = pulses;
 | 
			
		||||
  if (this->total_sensor_ != nullptr) {
 | 
			
		||||
    this->total_sensor_->publish_state(this->total_pulses_);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void PulseMeterSensor::setup() {
 | 
			
		||||
  this->pin_->setup();
 | 
			
		||||
  this->isr_pin_ = pin_->to_isr();
 | 
			
		||||
 
 | 
			
		||||
@@ -20,7 +20,8 @@ class PulseMeterSensor : public sensor::Sensor, public Component {
 | 
			
		||||
  void set_timeout_us(uint32_t timeout) { this->timeout_us_ = timeout; }
 | 
			
		||||
  void set_total_sensor(sensor::Sensor *sensor) { this->total_sensor_ = sensor; }
 | 
			
		||||
  void set_filter_mode(InternalFilterMode mode) { this->filter_mode_ = mode; }
 | 
			
		||||
  void set_total_pulses(uint32_t pulses) { this->total_pulses_ = pulses; }
 | 
			
		||||
 | 
			
		||||
  void set_total_pulses(uint32_t pulses);
 | 
			
		||||
 | 
			
		||||
  void setup() override;
 | 
			
		||||
  void loop() override;
 | 
			
		||||
 
 | 
			
		||||
@@ -19,7 +19,7 @@ from esphome.const import (
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import CORE
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@stevebaxter", "@cstaahl"]
 | 
			
		||||
CODEOWNERS = ["@stevebaxter", "@cstaahl", "@TrentHouliston"]
 | 
			
		||||
 | 
			
		||||
pulse_meter_ns = cg.esphome_ns.namespace("pulse_meter")
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -10,7 +10,7 @@ namespace socket {
 | 
			
		||||
Socket::~Socket() {}
 | 
			
		||||
 | 
			
		||||
std::unique_ptr<Socket> socket_ip(int type, int protocol) {
 | 
			
		||||
#if LWIP_IPV6
 | 
			
		||||
#if ENABLE_IPV6
 | 
			
		||||
  return socket(AF_INET6, type, protocol);
 | 
			
		||||
#else
 | 
			
		||||
  return socket(AF_INET, type, protocol);
 | 
			
		||||
@@ -18,7 +18,7 @@ std::unique_ptr<Socket> socket_ip(int type, int protocol) {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
socklen_t set_sockaddr(struct sockaddr *addr, socklen_t addrlen, const std::string &ip_address, uint16_t port) {
 | 
			
		||||
#if LWIP_IPV6
 | 
			
		||||
#if ENABLE_IPV6
 | 
			
		||||
  if (addrlen < sizeof(sockaddr_in6)) {
 | 
			
		||||
    errno = EINVAL;
 | 
			
		||||
    return 0;
 | 
			
		||||
@@ -51,7 +51,7 @@ socklen_t set_sockaddr(struct sockaddr *addr, socklen_t addrlen, const std::stri
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
socklen_t set_sockaddr_any(struct sockaddr *addr, socklen_t addrlen, uint16_t port) {
 | 
			
		||||
#if LWIP_IPV6
 | 
			
		||||
#if ENABLE_IPV6
 | 
			
		||||
  if (addrlen < sizeof(sockaddr_in6)) {
 | 
			
		||||
    errno = EINVAL;
 | 
			
		||||
    return 0;
 | 
			
		||||
 
 | 
			
		||||
@@ -316,8 +316,8 @@ void VoiceAssistant::loop() {
 | 
			
		||||
        this->speaker_buffer_index_ = 0;
 | 
			
		||||
        memset(this->speaker_buffer_, 0, SPEAKER_BUFFER_SIZE);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      this->wait_for_stream_end_ = false;
 | 
			
		||||
#endif
 | 
			
		||||
      this->set_state_(State::IDLE, State::IDLE);
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
@@ -586,7 +586,9 @@ void VoiceAssistant::on_event(const api::VoiceAssistantEventResponse &msg) {
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
    case api::enums::VOICE_ASSISTANT_TTS_STREAM_START: {
 | 
			
		||||
#ifdef USE_SPEAKER
 | 
			
		||||
      this->wait_for_stream_end_ = true;
 | 
			
		||||
#endif
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
    case api::enums::VOICE_ASSISTANT_TTS_STREAM_END: {
 | 
			
		||||
 
 | 
			
		||||
@@ -8,16 +8,16 @@
 | 
			
		||||
#include <user_interface.h>
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#include <utility>
 | 
			
		||||
#include <algorithm>
 | 
			
		||||
#include "lwip/err.h"
 | 
			
		||||
#include <utility>
 | 
			
		||||
#include "lwip/dns.h"
 | 
			
		||||
#include "lwip/err.h"
 | 
			
		||||
 | 
			
		||||
#include "esphome/core/application.h"
 | 
			
		||||
#include "esphome/core/hal.h"
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
#include "esphome/core/hal.h"
 | 
			
		||||
#include "esphome/core/util.h"
 | 
			
		||||
#include "esphome/core/application.h"
 | 
			
		||||
 | 
			
		||||
#ifdef USE_CAPTIVE_PORTAL
 | 
			
		||||
#include "esphome/components/captive_portal/captive_portal.h"
 | 
			
		||||
@@ -96,7 +96,7 @@ void WiFiComponent::start() {
 | 
			
		||||
#endif
 | 
			
		||||
  }
 | 
			
		||||
#ifdef USE_IMPROV
 | 
			
		||||
  if (esp32_improv::global_improv_component != nullptr) {
 | 
			
		||||
  if (!this->has_sta() && esp32_improv::global_improv_component != nullptr) {
 | 
			
		||||
    if (this->wifi_mode_(true, {}))
 | 
			
		||||
      esp32_improv::global_improv_component->start();
 | 
			
		||||
  }
 | 
			
		||||
@@ -163,8 +163,8 @@ void WiFiComponent::loop() {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
#ifdef USE_IMPROV
 | 
			
		||||
    if (esp32_improv::global_improv_component != nullptr) {
 | 
			
		||||
      if (!this->is_connected()) {
 | 
			
		||||
    if (esp32_improv::global_improv_component != nullptr && !esp32_improv::global_improv_component->is_active()) {
 | 
			
		||||
      if (now - this->last_connected_ > esp32_improv::global_improv_component->get_wifi_timeout()) {
 | 
			
		||||
        if (this->wifi_mode_(true, {}))
 | 
			
		||||
          esp32_improv::global_improv_component->start();
 | 
			
		||||
      }
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
"""Constants used by esphome."""
 | 
			
		||||
 | 
			
		||||
__version__ = "2023.10.0"
 | 
			
		||||
__version__ = "2023.10.2"
 | 
			
		||||
 | 
			
		||||
ALLOWED_NAME_CHARS = "abcdefghijklmnopqrstuvwxyz0123456789-_"
 | 
			
		||||
VALID_SUBSTITUTIONS_CHARACTERS = (
 | 
			
		||||
 
 | 
			
		||||
@@ -52,12 +52,12 @@ template<typename... Ts> class XorCondition : public Condition<Ts...> {
 | 
			
		||||
 public:
 | 
			
		||||
  explicit XorCondition(const std::vector<Condition<Ts...> *> &conditions) : conditions_(conditions) {}
 | 
			
		||||
  bool check(Ts... x) override {
 | 
			
		||||
    bool xor_state = false;
 | 
			
		||||
    size_t result = 0;
 | 
			
		||||
    for (auto *condition : this->conditions_) {
 | 
			
		||||
      xor_state = xor_state ^ condition->check(x...);
 | 
			
		||||
      result += condition->check(x...);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return xor_state;
 | 
			
		||||
    return result == 1;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
 
 | 
			
		||||
@@ -3326,6 +3326,10 @@ text_sensor:
 | 
			
		||||
          canbus_id: mcp2515_can
 | 
			
		||||
          can_id: 23
 | 
			
		||||
          data: [0x10, 0x20, 0x30]
 | 
			
		||||
      - canbus.send:
 | 
			
		||||
          canbus_id: mcp2515_can
 | 
			
		||||
          can_id: 23
 | 
			
		||||
          data: !lambda return {0x10, 0x20, 0x30};
 | 
			
		||||
      - canbus.send:
 | 
			
		||||
          canbus_id: esp32_internal_can
 | 
			
		||||
          can_id: 23
 | 
			
		||||
 
 | 
			
		||||
@@ -26,3 +26,9 @@ sensor:
 | 
			
		||||
    name: ADC
 | 
			
		||||
    pin: GPIO23
 | 
			
		||||
    update_interval: 1s
 | 
			
		||||
 | 
			
		||||
mqtt:
 | 
			
		||||
  broker: test.mosquitto.org
 | 
			
		||||
  port: 1883
 | 
			
		||||
  discovery: true
 | 
			
		||||
  discovery_prefix: homeassistant
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user