mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-26 04:33:47 +00:00 
			
		
		
		
	[improv_serial] Various optimizations (#11473)
Co-authored-by: J. Nick Koston <nick@koston.org>
This commit is contained in:
		| @@ -28,6 +28,38 @@ void ImprovSerialComponent::setup() { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | void ImprovSerialComponent::loop() { | ||||||
|  |   if (this->last_read_byte_ && (millis() - this->last_read_byte_ > IMPROV_SERIAL_TIMEOUT)) { | ||||||
|  |     this->last_read_byte_ = 0; | ||||||
|  |     this->rx_buffer_.clear(); | ||||||
|  |     ESP_LOGV(TAG, "Timeout"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   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) { | ||||||
|  |     if (wifi::global_wifi_component->is_connected()) { | ||||||
|  |       wifi::global_wifi_component->save_wifi_sta(this->connecting_sta_.get_ssid(), | ||||||
|  |                                                  this->connecting_sta_.get_password()); | ||||||
|  |       this->connecting_sta_ = {}; | ||||||
|  |       this->cancel_timeout("wifi-connect-timeout"); | ||||||
|  |       this->set_state_(improv::STATE_PROVISIONED); | ||||||
|  |  | ||||||
|  |       std::vector<uint8_t> url = this->build_rpc_settings_response_(improv::WIFI_SETTINGS); | ||||||
|  |       this->send_response_(url); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| void ImprovSerialComponent::dump_config() { ESP_LOGCONFIG(TAG, "Improv Serial:"); } | void ImprovSerialComponent::dump_config() { ESP_LOGCONFIG(TAG, "Improv Serial:"); } | ||||||
|  |  | ||||||
| optional<uint8_t> ImprovSerialComponent::read_byte_() { | optional<uint8_t> ImprovSerialComponent::read_byte_() { | ||||||
| @@ -78,8 +110,28 @@ optional<uint8_t> ImprovSerialComponent::read_byte_() { | |||||||
|   return byte; |   return byte; | ||||||
| } | } | ||||||
|  |  | ||||||
| void ImprovSerialComponent::write_data_(std::vector<uint8_t> &data) { | void ImprovSerialComponent::write_data_(const uint8_t *data, const size_t size) { | ||||||
|   data.push_back('\n'); |   // First, set length field | ||||||
|  |   this->tx_header_[TX_LENGTH_IDX] = this->tx_header_[TX_TYPE_IDX] == TYPE_RPC_RESPONSE ? size : 1; | ||||||
|  |  | ||||||
|  |   const bool there_is_data = data != nullptr && size > 0; | ||||||
|  |   // If there_is_data, checksum must not include our optional data byte | ||||||
|  |   const uint8_t header_checksum_len = there_is_data ? TX_BUFFER_SIZE - 3 : TX_BUFFER_SIZE - 2; | ||||||
|  |   // Only transmit the full buffer length if there is no data (only state/error byte is provided in this case) | ||||||
|  |   const uint8_t header_tx_len = there_is_data ? TX_BUFFER_SIZE - 3 : TX_BUFFER_SIZE; | ||||||
|  |   // Calculate checksum for message | ||||||
|  |   uint8_t checksum = 0; | ||||||
|  |   for (uint8_t i = 0; i < header_checksum_len; i++) { | ||||||
|  |     checksum += this->tx_header_[i]; | ||||||
|  |   } | ||||||
|  |   if (there_is_data) { | ||||||
|  |     // Include data in checksum | ||||||
|  |     for (size_t i = 0; i < size; i++) { | ||||||
|  |       checksum += data[i]; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   this->tx_header_[TX_CHECKSUM_IDX] = checksum; | ||||||
|  |  | ||||||
| #ifdef USE_ESP32 | #ifdef USE_ESP32 | ||||||
|   switch (logger::global_logger->get_uart()) { |   switch (logger::global_logger->get_uart()) { | ||||||
|     case logger::UART_SELECTION_UART0: |     case logger::UART_SELECTION_UART0: | ||||||
| @@ -87,63 +139,45 @@ void ImprovSerialComponent::write_data_(std::vector<uint8_t> &data) { | |||||||
| #if !defined(USE_ESP32_VARIANT_ESP32C3) && !defined(USE_ESP32_VARIANT_ESP32C6) && \ | #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_ESP32S2) && !defined(USE_ESP32_VARIANT_ESP32S3) | ||||||
|     case logger::UART_SELECTION_UART2: |     case logger::UART_SELECTION_UART2: | ||||||
| #endif  // !USE_ESP32_VARIANT_ESP32C3 && !USE_ESP32_VARIANT_ESP32S2 && !USE_ESP32_VARIANT_ESP32S3 | #endif | ||||||
|       uart_write_bytes(this->uart_num_, data.data(), data.size()); |       uart_write_bytes(this->uart_num_, this->tx_header_, header_tx_len); | ||||||
|  |       if (there_is_data) { | ||||||
|  |         uart_write_bytes(this->uart_num_, data, size); | ||||||
|  |         uart_write_bytes(this->uart_num_, &this->tx_header_[TX_CHECKSUM_IDX], 2);  // Footer: checksum and newline | ||||||
|  |       } | ||||||
|       break; |       break; | ||||||
| #if defined(USE_LOGGER_USB_CDC) && defined(CONFIG_ESP_CONSOLE_USB_CDC) | #if defined(USE_LOGGER_USB_CDC) && defined(CONFIG_ESP_CONSOLE_USB_CDC) | ||||||
|     case logger::UART_SELECTION_USB_CDC: { |     case logger::UART_SELECTION_USB_CDC: | ||||||
|       const char *msg = (char *) data.data(); |       esp_usb_console_write_buf((const char *) this->tx_header_, header_tx_len); | ||||||
|       esp_usb_console_write_buf(msg, data.size()); |       if (there_is_data) { | ||||||
|  |         esp_usb_console_write_buf((const char *) data, size); | ||||||
|  |         esp_usb_console_write_buf((const char *) &this->tx_header_[TX_CHECKSUM_IDX], | ||||||
|  |                                   2);  // Footer: checksum and newline | ||||||
|  |       } | ||||||
|       break; |       break; | ||||||
|     } | #endif | ||||||
| #endif  // USE_LOGGER_USB_CDC |  | ||||||
| #ifdef USE_LOGGER_USB_SERIAL_JTAG | #ifdef USE_LOGGER_USB_SERIAL_JTAG | ||||||
|     case logger::UART_SELECTION_USB_SERIAL_JTAG: |     case logger::UART_SELECTION_USB_SERIAL_JTAG: | ||||||
|       usb_serial_jtag_write_bytes((char *) data.data(), data.size(), 20 / portTICK_PERIOD_MS); |       usb_serial_jtag_write_bytes((const char *) this->tx_header_, header_tx_len, 20 / portTICK_PERIOD_MS); | ||||||
|       delay(10); |       if (there_is_data) { | ||||||
|       usb_serial_jtag_ll_txfifo_flush();  // fixes for issue in IDF 4.4.7 |         usb_serial_jtag_write_bytes((const char *) data, size, 20 / portTICK_PERIOD_MS); | ||||||
|  |         usb_serial_jtag_write_bytes((const char *) &this->tx_header_[TX_CHECKSUM_IDX], 2, | ||||||
|  |                                     20 / portTICK_PERIOD_MS);  // Footer: checksum and newline | ||||||
|  |       } | ||||||
|       break; |       break; | ||||||
| #endif  // USE_LOGGER_USB_SERIAL_JTAG | #endif | ||||||
|     default: |     default: | ||||||
|       break; |       break; | ||||||
|   } |   } | ||||||
| #elif defined(USE_ARDUINO) | #elif defined(USE_ARDUINO) | ||||||
|   this->hw_serial_->write(data.data(), data.size()); |   this->hw_serial_->write(this->tx_header_, header_tx_len); | ||||||
|  |   if (there_is_data) { | ||||||
|  |     this->hw_serial_->write(data, size); | ||||||
|  |     this->hw_serial_->write(&this->tx_header_[TX_CHECKSUM_IDX], 2);  // Footer: checksum and newline | ||||||
|  |   } | ||||||
| #endif | #endif | ||||||
| } | } | ||||||
|  |  | ||||||
| void ImprovSerialComponent::loop() { |  | ||||||
|   if (this->last_read_byte_ && (millis() - this->last_read_byte_ > IMPROV_SERIAL_TIMEOUT)) { |  | ||||||
|     this->last_read_byte_ = 0; |  | ||||||
|     this->rx_buffer_.clear(); |  | ||||||
|     ESP_LOGV(TAG, "Improv Serial timeout"); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   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) { |  | ||||||
|     if (wifi::global_wifi_component->is_connected()) { |  | ||||||
|       wifi::global_wifi_component->save_wifi_sta(this->connecting_sta_.get_ssid(), |  | ||||||
|                                                  this->connecting_sta_.get_password()); |  | ||||||
|       this->connecting_sta_ = {}; |  | ||||||
|       this->cancel_timeout("wifi-connect-timeout"); |  | ||||||
|       this->set_state_(improv::STATE_PROVISIONED); |  | ||||||
|  |  | ||||||
|       std::vector<uint8_t> url = this->build_rpc_settings_response_(improv::WIFI_SETTINGS); |  | ||||||
|       this->send_response_(url); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| std::vector<uint8_t> ImprovSerialComponent::build_rpc_settings_response_(improv::Command command) { | std::vector<uint8_t> ImprovSerialComponent::build_rpc_settings_response_(improv::Command command) { | ||||||
|   std::vector<std::string> urls; |   std::vector<std::string> urls; | ||||||
| #ifdef USE_IMPROV_SERIAL_NEXT_URL | #ifdef USE_IMPROV_SERIAL_NEXT_URL | ||||||
| @@ -177,13 +211,13 @@ std::vector<uint8_t> ImprovSerialComponent::build_version_info_() { | |||||||
| bool ImprovSerialComponent::parse_improv_serial_byte_(uint8_t byte) { | bool ImprovSerialComponent::parse_improv_serial_byte_(uint8_t byte) { | ||||||
|   size_t at = this->rx_buffer_.size(); |   size_t at = this->rx_buffer_.size(); | ||||||
|   this->rx_buffer_.push_back(byte); |   this->rx_buffer_.push_back(byte); | ||||||
|   ESP_LOGV(TAG, "Improv Serial byte: 0x%02X", byte); |   ESP_LOGV(TAG, "Byte: 0x%02X", byte); | ||||||
|   const uint8_t *raw = &this->rx_buffer_[0]; |   const uint8_t *raw = &this->rx_buffer_[0]; | ||||||
|  |  | ||||||
|   return improv::parse_improv_serial_byte( |   return improv::parse_improv_serial_byte( | ||||||
|       at, byte, raw, [this](improv::ImprovCommand command) -> bool { return this->parse_improv_payload_(command); }, |       at, byte, raw, [this](improv::ImprovCommand command) -> bool { return this->parse_improv_payload_(command); }, | ||||||
|       [this](improv::Error error) -> void { |       [this](improv::Error error) -> void { | ||||||
|         ESP_LOGW(TAG, "Error decoding Improv payload"); |         ESP_LOGW(TAG, "Error decoding payload"); | ||||||
|         this->set_error_(error); |         this->set_error_(error); | ||||||
|       }); |       }); | ||||||
| } | } | ||||||
| @@ -199,7 +233,7 @@ bool ImprovSerialComponent::parse_improv_payload_(improv::ImprovCommand &command | |||||||
|       wifi::global_wifi_component->set_sta(sta); |       wifi::global_wifi_component->set_sta(sta); | ||||||
|       wifi::global_wifi_component->start_connecting(sta, false); |       wifi::global_wifi_component->start_connecting(sta, false); | ||||||
|       this->set_state_(improv::STATE_PROVISIONING); |       this->set_state_(improv::STATE_PROVISIONING); | ||||||
|       ESP_LOGD(TAG, "Received Improv wifi settings ssid=%s, password=" LOG_SECRET("%s"), command.ssid.c_str(), |       ESP_LOGD(TAG, "Received settings: SSID=%s, password=" LOG_SECRET("%s"), command.ssid.c_str(), | ||||||
|                command.password.c_str()); |                command.password.c_str()); | ||||||
|  |  | ||||||
|       auto f = std::bind(&ImprovSerialComponent::on_wifi_connect_timeout_, this); |       auto f = std::bind(&ImprovSerialComponent::on_wifi_connect_timeout_, this); | ||||||
| @@ -240,7 +274,7 @@ bool ImprovSerialComponent::parse_improv_payload_(improv::ImprovCommand &command | |||||||
|       return true; |       return true; | ||||||
|     } |     } | ||||||
|     default: { |     default: { | ||||||
|       ESP_LOGW(TAG, "Unknown Improv payload"); |       ESP_LOGW(TAG, "Unknown payload"); | ||||||
|       this->set_error_(improv::ERROR_UNKNOWN_RPC); |       this->set_error_(improv::ERROR_UNKNOWN_RPC); | ||||||
|       return false; |       return false; | ||||||
|     } |     } | ||||||
| @@ -249,57 +283,26 @@ bool ImprovSerialComponent::parse_improv_payload_(improv::ImprovCommand &command | |||||||
|  |  | ||||||
| void ImprovSerialComponent::set_state_(improv::State state) { | void ImprovSerialComponent::set_state_(improv::State state) { | ||||||
|   this->state_ = state; |   this->state_ = state; | ||||||
|  |   this->tx_header_[TX_TYPE_IDX] = TYPE_CURRENT_STATE; | ||||||
|   std::vector<uint8_t> data = {'I', 'M', 'P', 'R', 'O', 'V'}; |   this->tx_header_[TX_DATA_IDX] = state; | ||||||
|   data.resize(11); |   this->write_data_(); | ||||||
|   data[6] = IMPROV_SERIAL_VERSION; |  | ||||||
|   data[7] = TYPE_CURRENT_STATE; |  | ||||||
|   data[8] = 1; |  | ||||||
|   data[9] = state; |  | ||||||
|  |  | ||||||
|   uint8_t checksum = 0x00; |  | ||||||
|   for (uint8_t d : data) |  | ||||||
|     checksum += d; |  | ||||||
|   data[10] = checksum; |  | ||||||
|  |  | ||||||
|   this->write_data_(data); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void ImprovSerialComponent::set_error_(improv::Error error) { | void ImprovSerialComponent::set_error_(improv::Error error) { | ||||||
|   std::vector<uint8_t> data = {'I', 'M', 'P', 'R', 'O', 'V'}; |   this->tx_header_[TX_TYPE_IDX] = TYPE_ERROR_STATE; | ||||||
|   data.resize(11); |   this->tx_header_[TX_DATA_IDX] = error; | ||||||
|   data[6] = IMPROV_SERIAL_VERSION; |   this->write_data_(); | ||||||
|   data[7] = TYPE_ERROR_STATE; |  | ||||||
|   data[8] = 1; |  | ||||||
|   data[9] = error; |  | ||||||
|  |  | ||||||
|   uint8_t checksum = 0x00; |  | ||||||
|   for (uint8_t d : data) |  | ||||||
|     checksum += d; |  | ||||||
|   data[10] = checksum; |  | ||||||
|   this->write_data_(data); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void ImprovSerialComponent::send_response_(std::vector<uint8_t> &response) { | void ImprovSerialComponent::send_response_(std::vector<uint8_t> &response) { | ||||||
|   std::vector<uint8_t> data = {'I', 'M', 'P', 'R', 'O', 'V'}; |   this->tx_header_[TX_TYPE_IDX] = TYPE_RPC_RESPONSE; | ||||||
|   data.resize(9); |   this->write_data_(response.data(), response.size()); | ||||||
|   data[6] = IMPROV_SERIAL_VERSION; |  | ||||||
|   data[7] = TYPE_RPC_RESPONSE; |  | ||||||
|   data[8] = response.size(); |  | ||||||
|   data.insert(data.end(), response.begin(), response.end()); |  | ||||||
|  |  | ||||||
|   uint8_t checksum = 0x00; |  | ||||||
|   for (uint8_t d : data) |  | ||||||
|     checksum += d; |  | ||||||
|   data.push_back(checksum); |  | ||||||
|  |  | ||||||
|   this->write_data_(data); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void ImprovSerialComponent::on_wifi_connect_timeout_() { | void ImprovSerialComponent::on_wifi_connect_timeout_() { | ||||||
|   this->set_error_(improv::ERROR_UNABLE_TO_CONNECT); |   this->set_error_(improv::ERROR_UNABLE_TO_CONNECT); | ||||||
|   this->set_state_(improv::STATE_AUTHORIZED); |   this->set_state_(improv::STATE_AUTHORIZED); | ||||||
|   ESP_LOGW(TAG, "Timed out trying to connect to given WiFi network"); |   ESP_LOGW(TAG, "Timed out while connecting to Wi-Fi network"); | ||||||
|   wifi::global_wifi_component->clear_sta(); |   wifi::global_wifi_component->clear_sta(); | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -26,6 +26,16 @@ | |||||||
| namespace esphome { | namespace esphome { | ||||||
| namespace improv_serial { | namespace improv_serial { | ||||||
|  |  | ||||||
|  | // TX buffer layout constants | ||||||
|  | static constexpr uint8_t TX_HEADER_SIZE = 6;  // Bytes 0-5 = "IMPROV" | ||||||
|  | static constexpr uint8_t TX_VERSION_IDX = 6; | ||||||
|  | static constexpr uint8_t TX_TYPE_IDX = 7; | ||||||
|  | static constexpr uint8_t TX_LENGTH_IDX = 8; | ||||||
|  | static constexpr uint8_t TX_DATA_IDX = 9;  // For state/error messages only | ||||||
|  | static constexpr uint8_t TX_CHECKSUM_IDX = 10; | ||||||
|  | static constexpr uint8_t TX_NEWLINE_IDX = 11; | ||||||
|  | static constexpr uint8_t TX_BUFFER_SIZE = 12; | ||||||
|  |  | ||||||
| enum ImprovSerialType : uint8_t { | enum ImprovSerialType : uint8_t { | ||||||
|   TYPE_CURRENT_STATE = 0x01, |   TYPE_CURRENT_STATE = 0x01, | ||||||
|   TYPE_ERROR_STATE = 0x02, |   TYPE_ERROR_STATE = 0x02, | ||||||
| @@ -57,7 +67,22 @@ class ImprovSerialComponent : public Component, public improv_base::ImprovBase { | |||||||
|   std::vector<uint8_t> build_version_info_(); |   std::vector<uint8_t> build_version_info_(); | ||||||
|  |  | ||||||
|   optional<uint8_t> read_byte_(); |   optional<uint8_t> read_byte_(); | ||||||
|   void write_data_(std::vector<uint8_t> &data); |   void write_data_(const uint8_t *data = nullptr, size_t size = 0); | ||||||
|  |  | ||||||
|  |   uint8_t tx_header_[TX_BUFFER_SIZE] = { | ||||||
|  |       'I',                    // 0: Header | ||||||
|  |       'M',                    // 1: Header | ||||||
|  |       'P',                    // 2: Header | ||||||
|  |       'R',                    // 3: Header | ||||||
|  |       'O',                    // 4: Header | ||||||
|  |       'V',                    // 5: Header | ||||||
|  |       IMPROV_SERIAL_VERSION,  // 6: Version | ||||||
|  |       0,                      // 7: ImprovSerialType | ||||||
|  |       0,                      // 8: Length | ||||||
|  |       0,                      // 9...X: Data (here, one byte reserved for state/error) | ||||||
|  |       0,                      // X + 10: Checksum | ||||||
|  |       '\n', | ||||||
|  |   }; | ||||||
|  |  | ||||||
| #ifdef USE_ESP32 | #ifdef USE_ESP32 | ||||||
|   uart_port_t uart_num_; |   uart_port_t uart_num_; | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								tests/components/improv_base/common-uart0.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								tests/components/improv_base/common-uart0.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | wifi: | ||||||
|  |   ssid: MySSID | ||||||
|  |   password: password1 | ||||||
|  |  | ||||||
|  | logger: | ||||||
|  |   hardware_uart: UART0 | ||||||
|  |  | ||||||
|  | improv_serial: | ||||||
							
								
								
									
										1
									
								
								tests/components/improv_base/test-uart0.esp8266-ard.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								tests/components/improv_base/test-uart0.esp8266-ard.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | <<: !include common-uart0.yaml | ||||||
		Reference in New Issue
	
	Block a user