mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-31 07:03:55 +00:00 
			
		
		
		
	Merge branch 'dev' into jesserockz-2025-457
This commit is contained in:
		| @@ -1 +1 @@ | ||||
| 4368db58e8f884aff245996b1e8b644cc0796c0bb2fa706d5740d40b823d3ac9 | ||||
| 499db61c1aa55b98b6629df603a56a1ba7aff5a9a7c781a5c1552a9dcd186c08 | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -466,7 +466,7 @@ jobs: | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON }} | ||||
|           cache-key: ${{ needs.common.outputs.cache-key }} | ||||
|       - uses: pre-commit/action@2c7b3805fd2a0fd8c1884dcaebf91fc102a13ecd # v3.0.1 | ||||
|       - uses: esphome/action@43cd1109c09c544d97196f7730ee5b2e0cc6d81e # v3.0.1 fork with pinned actions/cache | ||||
|         env: | ||||
|           SKIP: pylint,clang-tidy-hash | ||||
|       - uses: pre-commit-ci/lite-action@5d6cc0eb514c891a40562a58a8e71576c5c7fb43 # v1.1.0 | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							| @@ -58,7 +58,7 @@ jobs: | ||||
|  | ||||
|       # Initializes the CodeQL tools for scanning. | ||||
|       - name: Initialize CodeQL | ||||
|         uses: github/codeql-action/init@3599b3baa15b485a2e49ef411a7a4bb2452e7f93 # v3.30.5 | ||||
|         uses: github/codeql-action/init@64d10c13136e1c5bce3e5fbde8d4906eeaafc885 # v3.30.6 | ||||
|         with: | ||||
|           languages: ${{ matrix.language }} | ||||
|           build-mode: ${{ matrix.build-mode }} | ||||
| @@ -86,6 +86,6 @@ jobs: | ||||
|           exit 1 | ||||
|  | ||||
|       - name: Perform CodeQL Analysis | ||||
|         uses: github/codeql-action/analyze@3599b3baa15b485a2e49ef411a7a4bb2452e7f93 # v3.30.5 | ||||
|         uses: github/codeql-action/analyze@64d10c13136e1c5bce3e5fbde8d4906eeaafc885 # v3.30.6 | ||||
|         with: | ||||
|           category: "/language:${{matrix.language}}" | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							| @@ -19,7 +19,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Stale | ||||
|         uses: actions/stale@3a9db7e6a41a89f618792c92c0e97cc736e1b13f # v10.0.0 | ||||
|         uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0 | ||||
|         with: | ||||
|           debug-only: ${{ github.ref != 'refs/heads/dev' }} # Dry-run when not run on dev branch | ||||
|           remove-stale-when-updated: true | ||||
|   | ||||
| @@ -11,7 +11,7 @@ ci: | ||||
| repos: | ||||
|   - repo: https://github.com/astral-sh/ruff-pre-commit | ||||
|     # Ruff version. | ||||
|     rev: v0.13.2 | ||||
|     rev: v0.13.3 | ||||
|     hooks: | ||||
|       # Run the linter. | ||||
|       - id: ruff | ||||
|   | ||||
| @@ -160,7 +160,6 @@ esphome/components/esp_ldo/* @clydebarrow | ||||
| esphome/components/espnow/* @jesserockz | ||||
| esphome/components/ethernet_info/* @gtjadsonsantos | ||||
| esphome/components/event/* @nohat | ||||
| esphome/components/event_emitter/* @Rapsssito | ||||
| esphome/components/exposure_notifications/* @OttoWinter | ||||
| esphome/components/ezo/* @ssieb | ||||
| esphome/components/ezo_pmp/* @carlos-sarmiento | ||||
| @@ -257,6 +256,7 @@ esphome/components/libretiny_pwm/* @kuba2k2 | ||||
| esphome/components/light/* @esphome/core | ||||
| esphome/components/lightwaverf/* @max246 | ||||
| esphome/components/lilygo_t5_47/touchscreen/* @jesserockz | ||||
| esphome/components/lm75b/* @beormund | ||||
| esphome/components/ln882x/* @lamauny | ||||
| esphome/components/lock/* @esphome/core | ||||
| esphome/components/logger/* @esphome/core | ||||
|   | ||||
| @@ -14,9 +14,11 @@ from typing import Protocol | ||||
|  | ||||
| import argcomplete | ||||
|  | ||||
| # Note: Do not import modules from esphome.components here, as this would | ||||
| # cause them to be loaded before external components are processed, resulting | ||||
| # in the built-in version being used instead of the external component one. | ||||
| from esphome import const, writer, yaml_util | ||||
| import esphome.codegen as cg | ||||
| from esphome.components.mqtt import CONF_DISCOVER_IP | ||||
| from esphome.config import iter_component_configs, read_config, strip_default_ids | ||||
| from esphome.const import ( | ||||
|     ALLOWED_NAME_CHARS, | ||||
| @@ -240,6 +242,8 @@ def has_ota() -> bool: | ||||
|  | ||||
| def has_mqtt_ip_lookup() -> bool: | ||||
|     """Check if MQTT is available and IP lookup is supported.""" | ||||
|     from esphome.components.mqtt import CONF_DISCOVER_IP | ||||
|  | ||||
|     if CONF_MQTT not in CORE.config: | ||||
|         return False | ||||
|     # Default Enabled | ||||
|   | ||||
| @@ -26,12 +26,12 @@ uint32_t Animation::get_animation_frame_count() const { return this->animation_f | ||||
| int Animation::get_current_frame() const { return this->current_frame_; } | ||||
| void Animation::next_frame() { | ||||
|   this->current_frame_++; | ||||
|   if (loop_count_ && this->current_frame_ == loop_end_frame_ && | ||||
|   if (loop_count_ && static_cast<uint32_t>(this->current_frame_) == loop_end_frame_ && | ||||
|       (this->loop_current_iteration_ < loop_count_ || loop_count_ < 0)) { | ||||
|     this->current_frame_ = loop_start_frame_; | ||||
|     this->loop_current_iteration_++; | ||||
|   } | ||||
|   if (this->current_frame_ >= animation_frame_count_) { | ||||
|   if (static_cast<uint32_t>(this->current_frame_) >= animation_frame_count_) { | ||||
|     this->loop_current_iteration_ = 1; | ||||
|     this->current_frame_ = 0; | ||||
|   } | ||||
|   | ||||
| @@ -14,6 +14,7 @@ from esphome.const import ( | ||||
|     CONF_EVENT, | ||||
|     CONF_ID, | ||||
|     CONF_KEY, | ||||
|     CONF_MAX_CONNECTIONS, | ||||
|     CONF_ON_CLIENT_CONNECTED, | ||||
|     CONF_ON_CLIENT_DISCONNECTED, | ||||
|     CONF_ON_ERROR, | ||||
| @@ -68,7 +69,7 @@ CONF_CUSTOM_SERVICES = "custom_services" | ||||
| CONF_HOMEASSISTANT_SERVICES = "homeassistant_services" | ||||
| CONF_HOMEASSISTANT_STATES = "homeassistant_states" | ||||
| CONF_LISTEN_BACKLOG = "listen_backlog" | ||||
| CONF_MAX_CONNECTIONS = "max_connections" | ||||
| CONF_MAX_SEND_QUEUE = "max_send_queue" | ||||
|  | ||||
|  | ||||
| def validate_encryption_key(value): | ||||
| @@ -191,6 +192,19 @@ CONFIG_SCHEMA = cv.All( | ||||
|                 host=8,  # Abundant resources | ||||
|                 ln882x=8,  # Moderate RAM | ||||
|             ): cv.int_range(min=1, max=20), | ||||
|             # Maximum queued send buffers per connection before dropping connection | ||||
|             # Each buffer uses ~8-12 bytes overhead plus actual message size | ||||
|             # Platform defaults based on available RAM and typical message rates: | ||||
|             cv.SplitDefault( | ||||
|                 CONF_MAX_SEND_QUEUE, | ||||
|                 esp8266=5,  # Limited RAM, need to fail fast | ||||
|                 esp32=8,  # More RAM, can buffer more | ||||
|                 rp2040=5,  # Limited RAM | ||||
|                 bk72xx=8,  # Moderate RAM | ||||
|                 rtl87xx=8,  # Moderate RAM | ||||
|                 host=16,  # Abundant resources | ||||
|                 ln882x=8,  # Moderate RAM | ||||
|             ): cv.int_range(min=1, max=64), | ||||
|         } | ||||
|     ).extend(cv.COMPONENT_SCHEMA), | ||||
|     cv.rename_key(CONF_SERVICES, CONF_ACTIONS), | ||||
| @@ -213,6 +227,7 @@ async def to_code(config): | ||||
|         cg.add(var.set_listen_backlog(config[CONF_LISTEN_BACKLOG])) | ||||
|     if CONF_MAX_CONNECTIONS in config: | ||||
|         cg.add(var.set_max_connections(config[CONF_MAX_CONNECTIONS])) | ||||
|     cg.add_define("API_MAX_SEND_QUEUE", config[CONF_MAX_SEND_QUEUE]) | ||||
|  | ||||
|     # Set USE_API_SERVICES if any services are enabled | ||||
|     if config.get(CONF_ACTIONS) or config[CONF_CUSTOM_SERVICES]: | ||||
|   | ||||
| @@ -116,8 +116,7 @@ void APIConnection::start() { | ||||
|  | ||||
|   APIError err = this->helper_->init(); | ||||
|   if (err != APIError::OK) { | ||||
|     on_fatal_error(); | ||||
|     this->log_warning_(LOG_STR("Helper init failed"), err); | ||||
|     this->fatal_error_with_log_(LOG_STR("Helper init failed"), err); | ||||
|     return; | ||||
|   } | ||||
|   this->client_info_.peername = helper_->getpeername(); | ||||
| @@ -147,8 +146,7 @@ void APIConnection::loop() { | ||||
|  | ||||
|   APIError err = this->helper_->loop(); | ||||
|   if (err != APIError::OK) { | ||||
|     on_fatal_error(); | ||||
|     this->log_socket_operation_failed_(err); | ||||
|     this->fatal_error_with_log_(LOG_STR("Socket operation failed"), err); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
| @@ -163,17 +161,13 @@ void APIConnection::loop() { | ||||
|         // No more data available | ||||
|         break; | ||||
|       } else if (err != APIError::OK) { | ||||
|         on_fatal_error(); | ||||
|         this->log_warning_(LOG_STR("Reading failed"), err); | ||||
|         this->fatal_error_with_log_(LOG_STR("Reading failed"), err); | ||||
|         return; | ||||
|       } else { | ||||
|         this->last_traffic_ = now; | ||||
|         // read a packet | ||||
|         if (buffer.data_len > 0) { | ||||
|           this->read_message(buffer.data_len, buffer.type, &buffer.container[buffer.data_offset]); | ||||
|         } else { | ||||
|           this->read_message(0, buffer.type, nullptr); | ||||
|         } | ||||
|         this->read_message(buffer.data_len, buffer.type, | ||||
|                            buffer.data_len > 0 ? &buffer.container[buffer.data_offset] : nullptr); | ||||
|         if (this->flags_.remove) | ||||
|           return; | ||||
|       } | ||||
| @@ -205,7 +199,8 @@ void APIConnection::loop() { | ||||
|     // Disconnect if not responded within 2.5*keepalive | ||||
|     if (now - this->last_traffic_ > KEEPALIVE_DISCONNECT_TIMEOUT) { | ||||
|       on_fatal_error(); | ||||
|       ESP_LOGW(TAG, "%s is unresponsive; disconnecting", this->get_client_combined_info().c_str()); | ||||
|       ESP_LOGW(TAG, "%s (%s) is unresponsive; disconnecting", this->client_info_.name.c_str(), | ||||
|                this->client_info_.peername.c_str()); | ||||
|     } | ||||
|   } else if (now - this->last_traffic_ > KEEPALIVE_TIMEOUT_MS && !this->flags_.remove) { | ||||
|     // Only send ping if we're not disconnecting | ||||
| @@ -255,7 +250,7 @@ bool APIConnection::send_disconnect_response(const DisconnectRequest &msg) { | ||||
|   // remote initiated disconnect_client | ||||
|   // don't close yet, we still need to send the disconnect response | ||||
|   // close will happen on next loop | ||||
|   ESP_LOGD(TAG, "%s disconnected", this->get_client_combined_info().c_str()); | ||||
|   ESP_LOGD(TAG, "%s (%s) disconnected", this->client_info_.name.c_str(), this->client_info_.peername.c_str()); | ||||
|   this->flags_.next_close = true; | ||||
|   DisconnectResponse resp; | ||||
|   return this->send_message(resp, DisconnectResponse::MESSAGE_TYPE); | ||||
| @@ -1385,7 +1380,7 @@ void APIConnection::complete_authentication_() { | ||||
|   } | ||||
|  | ||||
|   this->flags_.connection_state = static_cast<uint8_t>(ConnectionState::AUTHENTICATED); | ||||
|   ESP_LOGD(TAG, "%s connected", this->get_client_combined_info().c_str()); | ||||
|   ESP_LOGD(TAG, "%s (%s) connected", this->client_info_.name.c_str(), this->client_info_.peername.c_str()); | ||||
| #ifdef USE_API_CLIENT_CONNECTED_TRIGGER | ||||
|   this->parent_->get_client_connected_trigger()->trigger(this->client_info_.name, this->client_info_.peername); | ||||
| #endif | ||||
| @@ -1394,6 +1389,11 @@ void APIConnection::complete_authentication_() { | ||||
|     this->send_time_request(); | ||||
|   } | ||||
| #endif | ||||
| #ifdef USE_ZWAVE_PROXY | ||||
|   if (zwave_proxy::global_zwave_proxy != nullptr) { | ||||
|     zwave_proxy::global_zwave_proxy->api_connection_authenticated(this); | ||||
|   } | ||||
| #endif | ||||
| } | ||||
|  | ||||
| bool APIConnection::send_hello_response(const HelloRequest &msg) { | ||||
| @@ -1586,8 +1586,7 @@ bool APIConnection::try_to_clear_buffer(bool log_out_of_space) { | ||||
|   delay(0); | ||||
|   APIError err = this->helper_->loop(); | ||||
|   if (err != APIError::OK) { | ||||
|     on_fatal_error(); | ||||
|     this->log_socket_operation_failed_(err); | ||||
|     this->fatal_error_with_log_(LOG_STR("Socket operation failed"), err); | ||||
|     return false; | ||||
|   } | ||||
|   if (this->helper_->can_write_without_blocking()) | ||||
| @@ -1606,8 +1605,7 @@ bool APIConnection::send_buffer(ProtoWriteBuffer buffer, uint8_t message_type) { | ||||
|   if (err == APIError::WOULD_BLOCK) | ||||
|     return false; | ||||
|   if (err != APIError::OK) { | ||||
|     on_fatal_error(); | ||||
|     this->log_warning_(LOG_STR("Packet write failed"), err); | ||||
|     this->fatal_error_with_log_(LOG_STR("Packet write failed"), err); | ||||
|     return false; | ||||
|   } | ||||
|   // Do not set last_traffic_ on send | ||||
| @@ -1616,12 +1614,12 @@ bool APIConnection::send_buffer(ProtoWriteBuffer buffer, uint8_t message_type) { | ||||
| #ifdef USE_API_PASSWORD | ||||
| void APIConnection::on_unauthenticated_access() { | ||||
|   this->on_fatal_error(); | ||||
|   ESP_LOGD(TAG, "%s access without authentication", this->get_client_combined_info().c_str()); | ||||
|   ESP_LOGD(TAG, "%s (%s) no authentication", this->client_info_.name.c_str(), this->client_info_.peername.c_str()); | ||||
| } | ||||
| #endif | ||||
| void APIConnection::on_no_setup_connection() { | ||||
|   this->on_fatal_error(); | ||||
|   ESP_LOGD(TAG, "%s access without full connection", this->get_client_combined_info().c_str()); | ||||
|   ESP_LOGD(TAG, "%s (%s) no connection setup", this->client_info_.name.c_str(), this->client_info_.peername.c_str()); | ||||
| } | ||||
| void APIConnection::on_fatal_error() { | ||||
|   this->helper_->close(); | ||||
| @@ -1793,8 +1791,7 @@ void APIConnection::process_batch_() { | ||||
|   APIError err = this->helper_->write_protobuf_packets(ProtoWriteBuffer{&shared_buf}, | ||||
|                                                        std::span<const PacketInfo>(packet_info, packet_count)); | ||||
|   if (err != APIError::OK && err != APIError::WOULD_BLOCK) { | ||||
|     on_fatal_error(); | ||||
|     this->log_warning_(LOG_STR("Batch write failed"), err); | ||||
|     this->fatal_error_with_log_(LOG_STR("Batch write failed"), err); | ||||
|   } | ||||
|  | ||||
| #ifdef HAS_PROTO_MESSAGE_DUMP | ||||
| @@ -1873,12 +1870,8 @@ void APIConnection::process_state_subscriptions_() { | ||||
| #endif  // USE_API_HOMEASSISTANT_STATES | ||||
|  | ||||
| void APIConnection::log_warning_(const LogString *message, APIError err) { | ||||
|   ESP_LOGW(TAG, "%s: %s %s errno=%d", this->get_client_combined_info().c_str(), LOG_STR_ARG(message), | ||||
|            LOG_STR_ARG(api_error_to_logstr(err)), errno); | ||||
| } | ||||
|  | ||||
| void APIConnection::log_socket_operation_failed_(APIError err) { | ||||
|   this->log_warning_(LOG_STR("Socket operation failed"), err); | ||||
|   ESP_LOGW(TAG, "%s (%s): %s %s errno=%d", this->client_info_.name.c_str(), this->client_info_.peername.c_str(), | ||||
|            LOG_STR_ARG(message), LOG_STR_ARG(api_error_to_logstr(err)), errno); | ||||
| } | ||||
|  | ||||
| }  // namespace esphome::api | ||||
|   | ||||
| @@ -19,14 +19,6 @@ namespace esphome::api { | ||||
| struct ClientInfo { | ||||
|   std::string name;      // Client name from Hello message | ||||
|   std::string peername;  // IP:port from socket | ||||
|  | ||||
|   std::string get_combined_info() const { | ||||
|     if (name == peername) { | ||||
|       // Before Hello message, both are the same | ||||
|       return name; | ||||
|     } | ||||
|     return name + " (" + peername + ")"; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| // Keepalive timeout in milliseconds | ||||
| @@ -281,7 +273,8 @@ class APIConnection final : public APIServerConnection { | ||||
|   bool try_to_clear_buffer(bool log_out_of_space); | ||||
|   bool send_buffer(ProtoWriteBuffer buffer, uint8_t message_type) override; | ||||
|  | ||||
|   std::string get_client_combined_info() const { return this->client_info_.get_combined_info(); } | ||||
|   const std::string &get_name() const { return this->client_info_.name; } | ||||
|   const std::string &get_peername() const { return this->client_info_.peername; } | ||||
|  | ||||
|  protected: | ||||
|   // Helper function to handle authentication completion | ||||
| @@ -742,8 +735,11 @@ class APIConnection final : public APIServerConnection { | ||||
|  | ||||
|   // Helper function to log API errors with errno | ||||
|   void log_warning_(const LogString *message, APIError err); | ||||
|   // Specific helper for duplicated error message | ||||
|   void log_socket_operation_failed_(APIError err); | ||||
|   // Helper to handle fatal errors with logging | ||||
|   inline void fatal_error_with_log_(const LogString *message, APIError err) { | ||||
|     this->on_fatal_error(); | ||||
|     this->log_warning_(message, err); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| }  // namespace esphome::api | ||||
|   | ||||
| @@ -13,7 +13,8 @@ namespace esphome::api { | ||||
|  | ||||
| static const char *const TAG = "api.frame_helper"; | ||||
|  | ||||
| #define HELPER_LOG(msg, ...) ESP_LOGVV(TAG, "%s: " msg, this->client_info_->get_combined_info().c_str(), ##__VA_ARGS__) | ||||
| #define HELPER_LOG(msg, ...) \ | ||||
|   ESP_LOGVV(TAG, "%s (%s): " msg, this->client_info_->name.c_str(), this->client_info_->peername.c_str(), ##__VA_ARGS__) | ||||
|  | ||||
| #ifdef HELPER_LOG_PACKETS | ||||
| #define LOG_PACKET_RECEIVED(buffer) ESP_LOGVV(TAG, "Received frame: %s", format_hex_pretty(buffer).c_str()) | ||||
| @@ -80,7 +81,7 @@ const LogString *api_error_to_logstr(APIError err) { | ||||
|  | ||||
| // Default implementation for loop - handles sending buffered data | ||||
| APIError APIFrameHelper::loop() { | ||||
|   if (!this->tx_buf_.empty()) { | ||||
|   if (this->tx_buf_count_ > 0) { | ||||
|     APIError err = try_send_tx_buf_(); | ||||
|     if (err != APIError::OK && err != APIError::WOULD_BLOCK) { | ||||
|       return err; | ||||
| @@ -102,9 +103,20 @@ APIError APIFrameHelper::handle_socket_write_error_() { | ||||
| // Helper method to buffer data from IOVs | ||||
| void APIFrameHelper::buffer_data_from_iov_(const struct iovec *iov, int iovcnt, uint16_t total_write_len, | ||||
|                                            uint16_t offset) { | ||||
|   SendBuffer buffer; | ||||
|   buffer.size = total_write_len - offset; | ||||
|   buffer.data = std::make_unique<uint8_t[]>(buffer.size); | ||||
|   // Check if queue is full | ||||
|   if (this->tx_buf_count_ >= API_MAX_SEND_QUEUE) { | ||||
|     HELPER_LOG("Send queue full (%u buffers), dropping connection", this->tx_buf_count_); | ||||
|     this->state_ = State::FAILED; | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   uint16_t buffer_size = total_write_len - offset; | ||||
|   auto &buffer = this->tx_buf_[this->tx_buf_tail_]; | ||||
|   buffer = std::make_unique<SendBuffer>(SendBuffer{ | ||||
|       .data = std::make_unique<uint8_t[]>(buffer_size), | ||||
|       .size = buffer_size, | ||||
|       .offset = 0, | ||||
|   }); | ||||
|  | ||||
|   uint16_t to_skip = offset; | ||||
|   uint16_t write_pos = 0; | ||||
| @@ -117,12 +129,15 @@ void APIFrameHelper::buffer_data_from_iov_(const struct iovec *iov, int iovcnt, | ||||
|       // Include this segment (partially or fully) | ||||
|       const uint8_t *src = reinterpret_cast<uint8_t *>(iov[i].iov_base) + to_skip; | ||||
|       uint16_t len = static_cast<uint16_t>(iov[i].iov_len) - to_skip; | ||||
|       std::memcpy(buffer.data.get() + write_pos, src, len); | ||||
|       std::memcpy(buffer->data.get() + write_pos, src, len); | ||||
|       write_pos += len; | ||||
|       to_skip = 0; | ||||
|     } | ||||
|   } | ||||
|   this->tx_buf_.push_back(std::move(buffer)); | ||||
|  | ||||
|   // Update circular buffer tracking | ||||
|   this->tx_buf_tail_ = (this->tx_buf_tail_ + 1) % API_MAX_SEND_QUEUE; | ||||
|   this->tx_buf_count_++; | ||||
| } | ||||
|  | ||||
| // This method writes data to socket or buffers it | ||||
| @@ -140,7 +155,7 @@ APIError APIFrameHelper::write_raw_(const struct iovec *iov, int iovcnt, uint16_ | ||||
| #endif | ||||
|  | ||||
|   // Try to send any existing buffered data first if there is any | ||||
|   if (!this->tx_buf_.empty()) { | ||||
|   if (this->tx_buf_count_ > 0) { | ||||
|     APIError send_result = try_send_tx_buf_(); | ||||
|     // If real error occurred (not just WOULD_BLOCK), return it | ||||
|     if (send_result != APIError::OK && send_result != APIError::WOULD_BLOCK) { | ||||
| @@ -149,7 +164,7 @@ APIError APIFrameHelper::write_raw_(const struct iovec *iov, int iovcnt, uint16_ | ||||
|  | ||||
|     // If there is still data in the buffer, we can't send, buffer | ||||
|     // the new data and return | ||||
|     if (!this->tx_buf_.empty()) { | ||||
|     if (this->tx_buf_count_ > 0) { | ||||
|       this->buffer_data_from_iov_(iov, iovcnt, total_write_len, 0); | ||||
|       return APIError::OK;  // Success, data buffered | ||||
|     } | ||||
| @@ -177,32 +192,31 @@ APIError APIFrameHelper::write_raw_(const struct iovec *iov, int iovcnt, uint16_ | ||||
| } | ||||
|  | ||||
| // Common implementation for trying to send buffered data | ||||
| // IMPORTANT: Caller MUST ensure tx_buf_ is not empty before calling this method | ||||
| // IMPORTANT: Caller MUST ensure tx_buf_count_ > 0 before calling this method | ||||
| APIError APIFrameHelper::try_send_tx_buf_() { | ||||
|   // Try to send from tx_buf - we assume it's not empty as it's the caller's responsibility to check | ||||
|   bool tx_buf_empty = false; | ||||
|   while (!tx_buf_empty) { | ||||
|   while (this->tx_buf_count_ > 0) { | ||||
|     // Get the first buffer in the queue | ||||
|     SendBuffer &front_buffer = this->tx_buf_.front(); | ||||
|     SendBuffer *front_buffer = this->tx_buf_[this->tx_buf_head_].get(); | ||||
|  | ||||
|     // Try to send the remaining data in this buffer | ||||
|     ssize_t sent = this->socket_->write(front_buffer.current_data(), front_buffer.remaining()); | ||||
|     ssize_t sent = this->socket_->write(front_buffer->current_data(), front_buffer->remaining()); | ||||
|  | ||||
|     if (sent == -1) { | ||||
|       return this->handle_socket_write_error_(); | ||||
|     } else if (sent == 0) { | ||||
|       // Nothing sent but not an error | ||||
|       return APIError::WOULD_BLOCK; | ||||
|     } else if (static_cast<uint16_t>(sent) < front_buffer.remaining()) { | ||||
|     } else if (static_cast<uint16_t>(sent) < front_buffer->remaining()) { | ||||
|       // Partially sent, update offset | ||||
|       // Cast to ensure no overflow issues with uint16_t | ||||
|       front_buffer.offset += static_cast<uint16_t>(sent); | ||||
|       front_buffer->offset += static_cast<uint16_t>(sent); | ||||
|       return APIError::WOULD_BLOCK;  // Stop processing more buffers if we couldn't send a complete buffer | ||||
|     } else { | ||||
|       // Buffer completely sent, remove it from the queue | ||||
|       this->tx_buf_.pop_front(); | ||||
|       // Update empty status for the loop condition | ||||
|       tx_buf_empty = this->tx_buf_.empty(); | ||||
|       this->tx_buf_[this->tx_buf_head_].reset(); | ||||
|       this->tx_buf_head_ = (this->tx_buf_head_ + 1) % API_MAX_SEND_QUEUE; | ||||
|       this->tx_buf_count_--; | ||||
|       // Continue loop to try sending the next buffer | ||||
|     } | ||||
|   } | ||||
|   | ||||
| @@ -1,7 +1,8 @@ | ||||
| #pragma once | ||||
| #include <array> | ||||
| #include <cstdint> | ||||
| #include <deque> | ||||
| #include <limits> | ||||
| #include <memory> | ||||
| #include <span> | ||||
| #include <utility> | ||||
| #include <vector> | ||||
| @@ -79,7 +80,7 @@ class APIFrameHelper { | ||||
|   virtual APIError init() = 0; | ||||
|   virtual APIError loop(); | ||||
|   virtual APIError read_packet(ReadPacketBuffer *buffer) = 0; | ||||
|   bool can_write_without_blocking() { return state_ == State::DATA && tx_buf_.empty(); } | ||||
|   bool can_write_without_blocking() { return this->state_ == State::DATA && this->tx_buf_count_ == 0; } | ||||
|   std::string getpeername() { return socket_->getpeername(); } | ||||
|   int getpeername(struct sockaddr *addr, socklen_t *addrlen) { return socket_->getpeername(addr, addrlen); } | ||||
|   APIError close() { | ||||
| @@ -161,7 +162,7 @@ class APIFrameHelper { | ||||
|   }; | ||||
|  | ||||
|   // Containers (size varies, but typically 12+ bytes on 32-bit) | ||||
|   std::deque<SendBuffer> tx_buf_; | ||||
|   std::array<std::unique_ptr<SendBuffer>, API_MAX_SEND_QUEUE> tx_buf_; | ||||
|   std::vector<struct iovec> reusable_iovs_; | ||||
|   std::vector<uint8_t> rx_buf_; | ||||
|  | ||||
| @@ -174,7 +175,10 @@ class APIFrameHelper { | ||||
|   State state_{State::INITIALIZE}; | ||||
|   uint8_t frame_header_padding_{0}; | ||||
|   uint8_t frame_footer_size_{0}; | ||||
|   // 5 bytes total, 3 bytes padding | ||||
|   uint8_t tx_buf_head_{0}; | ||||
|   uint8_t tx_buf_tail_{0}; | ||||
|   uint8_t tx_buf_count_{0}; | ||||
|   // 8 bytes total, 0 bytes padding | ||||
|  | ||||
|   // Common initialization for both plaintext and noise protocols | ||||
|   APIError init_common_(); | ||||
|   | ||||
| @@ -24,7 +24,8 @@ static const char *const PROLOGUE_INIT = "NoiseAPIInit"; | ||||
| #endif | ||||
| static constexpr size_t PROLOGUE_INIT_LEN = 12;  // strlen("NoiseAPIInit") | ||||
|  | ||||
| #define HELPER_LOG(msg, ...) ESP_LOGVV(TAG, "%s: " msg, this->client_info_->get_combined_info().c_str(), ##__VA_ARGS__) | ||||
| #define HELPER_LOG(msg, ...) \ | ||||
|   ESP_LOGVV(TAG, "%s (%s): " msg, this->client_info_->name.c_str(), this->client_info_->peername.c_str(), ##__VA_ARGS__) | ||||
|  | ||||
| #ifdef HELPER_LOG_PACKETS | ||||
| #define LOG_PACKET_RECEIVED(buffer) ESP_LOGVV(TAG, "Received frame: %s", format_hex_pretty(buffer).c_str()) | ||||
|   | ||||
| @@ -18,7 +18,8 @@ namespace esphome::api { | ||||
|  | ||||
| static const char *const TAG = "api.plaintext"; | ||||
|  | ||||
| #define HELPER_LOG(msg, ...) ESP_LOGVV(TAG, "%s: " msg, this->client_info_->get_combined_info().c_str(), ##__VA_ARGS__) | ||||
| #define HELPER_LOG(msg, ...) \ | ||||
|   ESP_LOGVV(TAG, "%s (%s): " msg, this->client_info_->name.c_str(), this->client_info_->peername.c_str(), ##__VA_ARGS__) | ||||
|  | ||||
| #ifdef HELPER_LOG_PACKETS | ||||
| #define LOG_PACKET_RECEIVED(buffer) ESP_LOGVV(TAG, "Received frame: %s", format_hex_pretty(buffer).c_str()) | ||||
|   | ||||
| @@ -181,7 +181,8 @@ void APIServer::loop() { | ||||
|     // Network is down - disconnect all clients | ||||
|     for (auto &client : this->clients_) { | ||||
|       client->on_fatal_error(); | ||||
|       ESP_LOGW(TAG, "%s: Network down; disconnect", client->get_client_combined_info().c_str()); | ||||
|       ESP_LOGW(TAG, "%s (%s): Network down; disconnect", client->client_info_.name.c_str(), | ||||
|                client->client_info_.peername.c_str()); | ||||
|     } | ||||
|     // Continue to process and clean up the clients below | ||||
|   } | ||||
|   | ||||
| @@ -55,7 +55,7 @@ template<typename... Ts> class UserServiceBase : public UserServiceDescriptor { | ||||
|  | ||||
|  protected: | ||||
|   virtual void execute(Ts... x) = 0; | ||||
|   template<int... S> void execute_(std::vector<ExecuteServiceArgument> args, seq<S...> type) { | ||||
|   template<int... S> void execute_(const std::vector<ExecuteServiceArgument> &args, seq<S...> type) { | ||||
|     this->execute((get_execute_arg_value<Ts>(args[S]))...); | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -57,7 +57,7 @@ const char *audio_file_type_to_string(AudioFileType file_type) { | ||||
| void scale_audio_samples(const int16_t *audio_samples, int16_t *output_buffer, int16_t scale_factor, | ||||
|                          size_t samples_to_scale) { | ||||
|   // Note the assembly dsps_mulc function has audio glitches if the input and output buffers are the same. | ||||
|   for (int i = 0; i < samples_to_scale; i++) { | ||||
|   for (size_t i = 0; i < samples_to_scale; i++) { | ||||
|     int32_t acc = (int32_t) audio_samples[i] * (int32_t) scale_factor; | ||||
|     output_buffer[i] = (int16_t) (acc >> 15); | ||||
|   } | ||||
|   | ||||
| @@ -97,10 +97,10 @@ void BL0906::handle_actions_() { | ||||
|     return; | ||||
|   } | ||||
|   ActionCallbackFuncPtr ptr_func = nullptr; | ||||
|   for (int i = 0; i < this->action_queue_.size(); i++) { | ||||
|   for (size_t i = 0; i < this->action_queue_.size(); i++) { | ||||
|     ptr_func = this->action_queue_[i]; | ||||
|     if (ptr_func) { | ||||
|       ESP_LOGI(TAG, "HandleActionCallback[%d]", i); | ||||
|       ESP_LOGI(TAG, "HandleActionCallback[%zu]", i); | ||||
|       (this->*ptr_func)(); | ||||
|     } | ||||
|   } | ||||
|   | ||||
| @@ -51,7 +51,7 @@ void BL0942::loop() { | ||||
|   if (!avail) { | ||||
|     return; | ||||
|   } | ||||
|   if (avail < sizeof(buffer)) { | ||||
|   if (static_cast<size_t>(avail) < sizeof(buffer)) { | ||||
|     if (!this->rx_start_) { | ||||
|       this->rx_start_ = millis(); | ||||
|     } else if (millis() > this->rx_start_ + PKT_TIMEOUT_MS) { | ||||
| @@ -148,7 +148,7 @@ void BL0942::setup() { | ||||
|  | ||||
|   this->write_reg_(BL0942_REG_USR_WRPROT, 0); | ||||
|  | ||||
|   if (this->read_reg_(BL0942_REG_MODE) != mode) | ||||
|   if (static_cast<uint32_t>(this->read_reg_(BL0942_REG_MODE)) != mode) | ||||
|     this->status_set_warning(LOG_STR("BL0942 setup failed!")); | ||||
|  | ||||
|   this->flush(); | ||||
|   | ||||
| @@ -116,7 +116,7 @@ CONFIG_SCHEMA = cv.All( | ||||
|     ) | ||||
|     .extend(cv.COMPONENT_SCHEMA) | ||||
|     .extend(esp32_ble_tracker.ESP_BLE_DEVICE_SCHEMA), | ||||
|     esp32_ble_tracker.consume_connection_slots(1, "ble_client"), | ||||
|     esp32_ble.consume_connection_slots(1, "ble_client"), | ||||
| ) | ||||
|  | ||||
| CONF_BLE_CLIENT_ID = "ble_client_id" | ||||
|   | ||||
| @@ -42,9 +42,7 @@ def validate_connections(config): | ||||
|             ) | ||||
|     elif config[CONF_ACTIVE]: | ||||
|         connection_slots: int = config[CONF_CONNECTION_SLOTS] | ||||
|         esp32_ble_tracker.consume_connection_slots(connection_slots, "bluetooth_proxy")( | ||||
|             config | ||||
|         ) | ||||
|         esp32_ble.consume_connection_slots(connection_slots, "bluetooth_proxy")(config) | ||||
|  | ||||
|         return { | ||||
|             **config, | ||||
| @@ -65,11 +63,11 @@ CONFIG_SCHEMA = cv.All( | ||||
|                     default=DEFAULT_CONNECTION_SLOTS, | ||||
|                 ): cv.All( | ||||
|                     cv.positive_int, | ||||
|                     cv.Range(min=1, max=esp32_ble_tracker.IDF_MAX_CONNECTIONS), | ||||
|                     cv.Range(min=1, max=esp32_ble.IDF_MAX_CONNECTIONS), | ||||
|                 ), | ||||
|                 cv.Optional(CONF_CONNECTIONS): cv.All( | ||||
|                     cv.ensure_list(CONNECTION_SCHEMA), | ||||
|                     cv.Length(min=1, max=esp32_ble_tracker.IDF_MAX_CONNECTIONS), | ||||
|                     cv.Length(min=1, max=esp32_ble.IDF_MAX_CONNECTIONS), | ||||
|                 ), | ||||
|             } | ||||
|         ) | ||||
|   | ||||
| @@ -11,14 +11,14 @@ namespace captive_portal { | ||||
| static const char *const TAG = "captive_portal"; | ||||
|  | ||||
| void CaptivePortal::handle_config(AsyncWebServerRequest *request) { | ||||
|   AsyncResponseStream *stream = request->beginResponseStream(F("application/json")); | ||||
|   stream->addHeader(F("cache-control"), F("public, max-age=0, must-revalidate")); | ||||
|   AsyncResponseStream *stream = request->beginResponseStream(ESPHOME_F("application/json")); | ||||
|   stream->addHeader(ESPHOME_F("cache-control"), ESPHOME_F("public, max-age=0, must-revalidate")); | ||||
| #ifdef USE_ESP8266 | ||||
|   stream->print(F("{\"mac\":\"")); | ||||
|   stream->print(ESPHOME_F("{\"mac\":\"")); | ||||
|   stream->print(get_mac_address_pretty().c_str()); | ||||
|   stream->print(F("\",\"name\":\"")); | ||||
|   stream->print(ESPHOME_F("\",\"name\":\"")); | ||||
|   stream->print(App.get_name().c_str()); | ||||
|   stream->print(F("\",\"aps\":[{}")); | ||||
|   stream->print(ESPHOME_F("\",\"aps\":[{}")); | ||||
| #else | ||||
|   stream->printf(R"({"mac":"%s","name":"%s","aps":[{})", get_mac_address_pretty().c_str(), App.get_name().c_str()); | ||||
| #endif | ||||
| @@ -29,19 +29,19 @@ void CaptivePortal::handle_config(AsyncWebServerRequest *request) { | ||||
|  | ||||
|       // Assumes no " in ssid, possible unicode isses? | ||||
| #ifdef USE_ESP8266 | ||||
|     stream->print(F(",{\"ssid\":\"")); | ||||
|     stream->print(ESPHOME_F(",{\"ssid\":\"")); | ||||
|     stream->print(scan.get_ssid().c_str()); | ||||
|     stream->print(F("\",\"rssi\":")); | ||||
|     stream->print(ESPHOME_F("\",\"rssi\":")); | ||||
|     stream->print(scan.get_rssi()); | ||||
|     stream->print(F(",\"lock\":")); | ||||
|     stream->print(ESPHOME_F(",\"lock\":")); | ||||
|     stream->print(scan.get_with_auth()); | ||||
|     stream->print(F("}")); | ||||
|     stream->print(ESPHOME_F("}")); | ||||
| #else | ||||
|     stream->printf(R"(,{"ssid":"%s","rssi":%d,"lock":%d})", scan.get_ssid().c_str(), scan.get_rssi(), | ||||
|                    scan.get_with_auth()); | ||||
| #endif | ||||
|   } | ||||
|   stream->print(F("]}")); | ||||
|   stream->print(ESPHOME_F("]}")); | ||||
|   request->send(stream); | ||||
| } | ||||
| void CaptivePortal::handle_wifisave(AsyncWebServerRequest *request) { | ||||
| @@ -52,7 +52,7 @@ void CaptivePortal::handle_wifisave(AsyncWebServerRequest *request) { | ||||
|   ESP_LOGI(TAG, "  Password=" LOG_SECRET("'%s'"), psk.c_str()); | ||||
|   wifi::global_wifi_component->save_wifi_sta(ssid, psk); | ||||
|   wifi::global_wifi_component->start_scanning(); | ||||
|   request->redirect(F("/?save")); | ||||
|   request->redirect(ESPHOME_F("/?save")); | ||||
| } | ||||
|  | ||||
| void CaptivePortal::setup() { | ||||
| @@ -75,7 +75,7 @@ void CaptivePortal::start() { | ||||
| #ifdef USE_ARDUINO | ||||
|   this->dns_server_ = make_unique<DNSServer>(); | ||||
|   this->dns_server_->setErrorReplyCode(DNSReplyCode::NoError); | ||||
|   this->dns_server_->start(53, F("*"), ip); | ||||
|   this->dns_server_->start(53, ESPHOME_F("*"), ip); | ||||
| #endif | ||||
|  | ||||
|   this->initialized_ = true; | ||||
| @@ -88,10 +88,10 @@ void CaptivePortal::start() { | ||||
| } | ||||
|  | ||||
| void CaptivePortal::handleRequest(AsyncWebServerRequest *req) { | ||||
|   if (req->url() == F("/config.json")) { | ||||
|   if (req->url() == ESPHOME_F("/config.json")) { | ||||
|     this->handle_config(req); | ||||
|     return; | ||||
|   } else if (req->url() == F("/wifisave")) { | ||||
|   } else if (req->url() == ESPHOME_F("/wifisave")) { | ||||
|     this->handle_wifisave(req); | ||||
|     return; | ||||
|   } | ||||
| @@ -100,11 +100,11 @@ void CaptivePortal::handleRequest(AsyncWebServerRequest *req) { | ||||
|   // This includes OS captive portal detection endpoints which will trigger | ||||
|   // the captive portal when they don't receive their expected responses | ||||
| #ifndef USE_ESP8266 | ||||
|   auto *response = req->beginResponse(200, F("text/html"), INDEX_GZ, sizeof(INDEX_GZ)); | ||||
|   auto *response = req->beginResponse(200, ESPHOME_F("text/html"), INDEX_GZ, sizeof(INDEX_GZ)); | ||||
| #else | ||||
|   auto *response = req->beginResponse_P(200, F("text/html"), INDEX_GZ, sizeof(INDEX_GZ)); | ||||
|   auto *response = req->beginResponse_P(200, ESPHOME_F("text/html"), INDEX_GZ, sizeof(INDEX_GZ)); | ||||
| #endif | ||||
|   response->addHeader(F("Content-Encoding"), F("gzip")); | ||||
|   response->addHeader(ESPHOME_F("Content-Encoding"), ESPHOME_F("gzip")); | ||||
|   req->send(response); | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -13,7 +13,7 @@ static const uint8_t C_M1106_CMD_SET_CO2_CALIB_RESPONSE[4] = {0x16, 0x01, 0x03, | ||||
|  | ||||
| uint8_t cm1106_checksum(const uint8_t *response, size_t len) { | ||||
|   uint8_t crc = 0; | ||||
|   for (int i = 0; i < len - 1; i++) { | ||||
|   for (size_t i = 0; i < len - 1; i++) { | ||||
|     crc -= response[i]; | ||||
|   } | ||||
|   return crc; | ||||
|   | ||||
| @@ -11,7 +11,7 @@ void CopyLock::setup() { | ||||
|  | ||||
|   traits.set_assumed_state(source_->traits.get_assumed_state()); | ||||
|   traits.set_requires_code(source_->traits.get_requires_code()); | ||||
|   traits.set_supported_states(source_->traits.get_supported_states()); | ||||
|   traits.set_supported_states_mask(source_->traits.get_supported_states_mask()); | ||||
|   traits.set_supports_open(source_->traits.get_supports_open()); | ||||
|  | ||||
|   this->publish_state(source_->state); | ||||
|   | ||||
| @@ -26,7 +26,7 @@ void DaikinArcClimate::transmit_query_() { | ||||
|   uint8_t remote_header[8] = {0x11, 0xDA, 0x27, 0x00, 0x84, 0x87, 0x20, 0x00}; | ||||
|  | ||||
|   // Calculate checksum | ||||
|   for (int i = 0; i < sizeof(remote_header) - 1; i++) { | ||||
|   for (size_t i = 0; i < sizeof(remote_header) - 1; i++) { | ||||
|     remote_header[sizeof(remote_header) - 1] += remote_header[i]; | ||||
|   } | ||||
|  | ||||
| @@ -102,7 +102,7 @@ void DaikinArcClimate::transmit_state() { | ||||
|   remote_state[9] = fan_speed & 0xff; | ||||
|  | ||||
|   // Calculate checksum | ||||
|   for (int i = 0; i < sizeof(remote_header) - 1; i++) { | ||||
|   for (size_t i = 0; i < sizeof(remote_header) - 1; i++) { | ||||
|     remote_header[sizeof(remote_header) - 1] += remote_header[i]; | ||||
|   } | ||||
|  | ||||
| @@ -350,7 +350,7 @@ bool DaikinArcClimate::on_receive(remote_base::RemoteReceiveData data) { | ||||
|   bool valid_daikin_frame = false; | ||||
|   if (data.expect_item(DAIKIN_HEADER_MARK, DAIKIN_HEADER_SPACE)) { | ||||
|     valid_daikin_frame = true; | ||||
|     int bytes_count = data.size() / 2 / 8; | ||||
|     size_t bytes_count = data.size() / 2 / 8; | ||||
|     std::unique_ptr<char[]> buf(new char[bytes_count * 3 + 1]); | ||||
|     buf[0] = '\0'; | ||||
|     for (size_t i = 0; i < bytes_count; i++) { | ||||
| @@ -370,7 +370,7 @@ bool DaikinArcClimate::on_receive(remote_base::RemoteReceiveData data) { | ||||
|   if (!valid_daikin_frame) { | ||||
|     char sbuf[16 * 10 + 1]; | ||||
|     sbuf[0] = '\0'; | ||||
|     for (size_t j = 0; j < data.size(); j++) { | ||||
|     for (size_t j = 0; j < static_cast<size_t>(data.size()); j++) { | ||||
|       if ((j - 2) % 16 == 0) { | ||||
|         if (j > 0) { | ||||
|           ESP_LOGD(TAG, "DATA %04x: %s", (j - 16 > 0xffff ? 0 : j - 16), sbuf); | ||||
| @@ -380,19 +380,26 @@ bool DaikinArcClimate::on_receive(remote_base::RemoteReceiveData data) { | ||||
|       char type_ch = ' '; | ||||
|       // debug_tolerance = 25% | ||||
|  | ||||
|       if (DAIKIN_DBG_LOWER(DAIKIN_ARC_PRE_MARK) <= data[j] && data[j] <= DAIKIN_DBG_UPPER(DAIKIN_ARC_PRE_MARK)) | ||||
|       if (static_cast<int32_t>(DAIKIN_DBG_LOWER(DAIKIN_ARC_PRE_MARK)) <= data[j] && | ||||
|           data[j] <= static_cast<int32_t>(DAIKIN_DBG_UPPER(DAIKIN_ARC_PRE_MARK))) | ||||
|         type_ch = 'P'; | ||||
|       if (DAIKIN_DBG_LOWER(DAIKIN_ARC_PRE_SPACE) <= -data[j] && -data[j] <= DAIKIN_DBG_UPPER(DAIKIN_ARC_PRE_SPACE)) | ||||
|       if (static_cast<int32_t>(DAIKIN_DBG_LOWER(DAIKIN_ARC_PRE_SPACE)) <= -data[j] && | ||||
|           -data[j] <= static_cast<int32_t>(DAIKIN_DBG_UPPER(DAIKIN_ARC_PRE_SPACE))) | ||||
|         type_ch = 'a'; | ||||
|       if (DAIKIN_DBG_LOWER(DAIKIN_HEADER_MARK) <= data[j] && data[j] <= DAIKIN_DBG_UPPER(DAIKIN_HEADER_MARK)) | ||||
|       if (static_cast<int32_t>(DAIKIN_DBG_LOWER(DAIKIN_HEADER_MARK)) <= data[j] && | ||||
|           data[j] <= static_cast<int32_t>(DAIKIN_DBG_UPPER(DAIKIN_HEADER_MARK))) | ||||
|         type_ch = 'H'; | ||||
|       if (DAIKIN_DBG_LOWER(DAIKIN_HEADER_SPACE) <= -data[j] && -data[j] <= DAIKIN_DBG_UPPER(DAIKIN_HEADER_SPACE)) | ||||
|       if (static_cast<int32_t>(DAIKIN_DBG_LOWER(DAIKIN_HEADER_SPACE)) <= -data[j] && | ||||
|           -data[j] <= static_cast<int32_t>(DAIKIN_DBG_UPPER(DAIKIN_HEADER_SPACE))) | ||||
|         type_ch = 'h'; | ||||
|       if (DAIKIN_DBG_LOWER(DAIKIN_BIT_MARK) <= data[j] && data[j] <= DAIKIN_DBG_UPPER(DAIKIN_BIT_MARK)) | ||||
|       if (static_cast<int32_t>(DAIKIN_DBG_LOWER(DAIKIN_BIT_MARK)) <= data[j] && | ||||
|           data[j] <= static_cast<int32_t>(DAIKIN_DBG_UPPER(DAIKIN_BIT_MARK))) | ||||
|         type_ch = 'B'; | ||||
|       if (DAIKIN_DBG_LOWER(DAIKIN_ONE_SPACE) <= -data[j] && -data[j] <= DAIKIN_DBG_UPPER(DAIKIN_ONE_SPACE)) | ||||
|       if (static_cast<int32_t>(DAIKIN_DBG_LOWER(DAIKIN_ONE_SPACE)) <= -data[j] && | ||||
|           -data[j] <= static_cast<int32_t>(DAIKIN_DBG_UPPER(DAIKIN_ONE_SPACE))) | ||||
|         type_ch = '1'; | ||||
|       if (DAIKIN_DBG_LOWER(DAIKIN_ZERO_SPACE) <= -data[j] && -data[j] <= DAIKIN_DBG_UPPER(DAIKIN_ZERO_SPACE)) | ||||
|       if (static_cast<int32_t>(DAIKIN_DBG_LOWER(DAIKIN_ZERO_SPACE)) <= -data[j] && | ||||
|           -data[j] <= static_cast<int32_t>(DAIKIN_DBG_UPPER(DAIKIN_ZERO_SPACE))) | ||||
|         type_ch = '0'; | ||||
|  | ||||
|       if (abs(data[j]) > 100000) { | ||||
| @@ -400,7 +407,7 @@ bool DaikinArcClimate::on_receive(remote_base::RemoteReceiveData data) { | ||||
|       } else { | ||||
|         sprintf(sbuf, "%s%-5d[%c] ", sbuf, (int) (round(data[j] / 10.) * 10), type_ch); | ||||
|       } | ||||
|       if (j == data.size() - 1) { | ||||
|       if (j + 1 == static_cast<size_t>(data.size())) { | ||||
|         ESP_LOGD(TAG, "DATA %04x: %s", (j - 8 > 0xffff ? 0 : j - 8), sbuf); | ||||
|       } | ||||
|     } | ||||
|   | ||||
| @@ -97,12 +97,12 @@ bool ES7210::set_mic_gain(float mic_gain) { | ||||
| } | ||||
|  | ||||
| bool ES7210::configure_sample_rate_() { | ||||
|   int mclk_fre = this->sample_rate_ * MCLK_DIV_FRE; | ||||
|   uint32_t mclk_fre = this->sample_rate_ * MCLK_DIV_FRE; | ||||
|   int coeff = -1; | ||||
|  | ||||
|   for (int i = 0; i < (sizeof(ES7210_COEFFICIENTS) / sizeof(ES7210_COEFFICIENTS[0])); ++i) { | ||||
|   for (size_t i = 0; i < (sizeof(ES7210_COEFFICIENTS) / sizeof(ES7210_COEFFICIENTS[0])); ++i) { | ||||
|     if (ES7210_COEFFICIENTS[i].lrclk == this->sample_rate_ && ES7210_COEFFICIENTS[i].mclk == mclk_fre) | ||||
|       coeff = i; | ||||
|       coeff = static_cast<int>(i); | ||||
|   } | ||||
|  | ||||
|   if (coeff >= 0) { | ||||
|   | ||||
| @@ -296,14 +296,9 @@ def _format_framework_arduino_version(ver: cv.Version) -> str: | ||||
|     return f"pioarduino/framework-arduinoespressif32@https://github.com/espressif/arduino-esp32/releases/download/{str(ver)}/esp32-{str(ver)}.zip" | ||||
|  | ||||
|  | ||||
| def _format_framework_espidf_version( | ||||
|     ver: cv.Version, release: str, for_platformio: bool | ||||
| ) -> str: | ||||
|     # format the given arduino (https://github.com/espressif/esp-idf/releases) version to | ||||
| def _format_framework_espidf_version(ver: cv.Version, release: str) -> str: | ||||
|     # format the given espidf (https://github.com/pioarduino/esp-idf/releases) version to | ||||
|     # a PIO platformio/framework-espidf value | ||||
|     # List of package versions: https://api.registry.platformio.org/v3/packages/platformio/tool/framework-espidf | ||||
|     if for_platformio: | ||||
|         return f"platformio/framework-espidf@~3.{ver.major}{ver.minor:02d}{ver.patch:02d}.0" | ||||
|     if release: | ||||
|         return f"pioarduino/framework-espidf@https://github.com/pioarduino/esp-idf/releases/download/v{str(ver)}.{release}/esp-idf-v{str(ver)}.zip" | ||||
|     return f"pioarduino/framework-espidf@https://github.com/pioarduino/esp-idf/releases/download/v{str(ver)}/esp-idf-v{str(ver)}.zip" | ||||
| @@ -317,157 +312,108 @@ def _format_framework_espidf_version( | ||||
|  | ||||
| # The default/recommended arduino framework version | ||||
| #  - https://github.com/espressif/arduino-esp32/releases | ||||
| RECOMMENDED_ARDUINO_FRAMEWORK_VERSION = cv.Version(3, 2, 1) | ||||
| # The platform-espressif32 version to use for arduino frameworks | ||||
| #  - https://github.com/pioarduino/platform-espressif32/releases | ||||
| ARDUINO_PLATFORM_VERSION = cv.Version(54, 3, 21, "2") | ||||
| ARDUINO_FRAMEWORK_VERSION_LOOKUP = { | ||||
|     "recommended": cv.Version(3, 2, 1), | ||||
|     "latest": cv.Version(3, 3, 1), | ||||
|     "dev": cv.Version(3, 3, 1), | ||||
| } | ||||
| ARDUINO_PLATFORM_VERSION_LOOKUP = { | ||||
|     cv.Version(3, 3, 1): cv.Version(55, 3, 31), | ||||
|     cv.Version(3, 3, 0): cv.Version(55, 3, 30, "2"), | ||||
|     cv.Version(3, 2, 1): cv.Version(54, 3, 21, "2"), | ||||
|     cv.Version(3, 2, 0): cv.Version(54, 3, 20), | ||||
|     cv.Version(3, 1, 3): cv.Version(53, 3, 13), | ||||
|     cv.Version(3, 1, 2): cv.Version(53, 3, 12), | ||||
|     cv.Version(3, 1, 1): cv.Version(53, 3, 11), | ||||
|     cv.Version(3, 1, 0): cv.Version(53, 3, 10), | ||||
| } | ||||
|  | ||||
| # The default/recommended esp-idf framework version | ||||
| #  - https://github.com/espressif/esp-idf/releases | ||||
| #  - https://api.registry.platformio.org/v3/packages/platformio/tool/framework-espidf | ||||
| RECOMMENDED_ESP_IDF_FRAMEWORK_VERSION = cv.Version(5, 4, 2) | ||||
| # The platformio/espressif32 version to use for esp-idf frameworks | ||||
| #  - https://github.com/platformio/platform-espressif32/releases | ||||
| #  - https://api.registry.platformio.org/v3/packages/platformio/platform/espressif32 | ||||
| ESP_IDF_PLATFORM_VERSION = cv.Version(54, 3, 21, "2") | ||||
| ESP_IDF_FRAMEWORK_VERSION_LOOKUP = { | ||||
|     "recommended": cv.Version(5, 4, 2), | ||||
|     "latest": cv.Version(5, 5, 1), | ||||
|     "dev": cv.Version(5, 5, 1), | ||||
| } | ||||
| ESP_IDF_PLATFORM_VERSION_LOOKUP = { | ||||
|     cv.Version(5, 5, 1): cv.Version(55, 3, 31), | ||||
|     cv.Version(5, 5, 0): cv.Version(55, 3, 31), | ||||
|     cv.Version(5, 4, 2): cv.Version(54, 3, 21, "2"), | ||||
|     cv.Version(5, 4, 1): cv.Version(54, 3, 21, "2"), | ||||
|     cv.Version(5, 4, 0): cv.Version(54, 3, 21, "2"), | ||||
|     cv.Version(5, 3, 2): cv.Version(53, 3, 13), | ||||
|     cv.Version(5, 3, 1): cv.Version(53, 3, 13), | ||||
|     cv.Version(5, 3, 0): cv.Version(53, 3, 13), | ||||
|     cv.Version(5, 1, 6): cv.Version(51, 3, 7), | ||||
|     cv.Version(5, 1, 5): cv.Version(51, 3, 7), | ||||
| } | ||||
|  | ||||
| # List based on https://registry.platformio.org/tools/platformio/framework-espidf/versions | ||||
| SUPPORTED_PLATFORMIO_ESP_IDF_5X = [ | ||||
|     cv.Version(5, 3, 1), | ||||
|     cv.Version(5, 3, 0), | ||||
|     cv.Version(5, 2, 2), | ||||
|     cv.Version(5, 2, 1), | ||||
|     cv.Version(5, 1, 2), | ||||
|     cv.Version(5, 1, 1), | ||||
|     cv.Version(5, 1, 0), | ||||
|     cv.Version(5, 0, 2), | ||||
|     cv.Version(5, 0, 1), | ||||
|     cv.Version(5, 0, 0), | ||||
| ] | ||||
|  | ||||
| # pioarduino versions that don't require a release number | ||||
| # List based on https://github.com/pioarduino/esp-idf/releases | ||||
| SUPPORTED_PIOARDUINO_ESP_IDF_5X = [ | ||||
|     cv.Version(5, 5, 1), | ||||
|     cv.Version(5, 5, 0), | ||||
|     cv.Version(5, 4, 2), | ||||
|     cv.Version(5, 4, 1), | ||||
|     cv.Version(5, 4, 0), | ||||
|     cv.Version(5, 3, 3), | ||||
|     cv.Version(5, 3, 2), | ||||
|     cv.Version(5, 3, 1), | ||||
|     cv.Version(5, 3, 0), | ||||
|     cv.Version(5, 1, 5), | ||||
|     cv.Version(5, 1, 6), | ||||
| ] | ||||
| # The platform-espressif32 version | ||||
| #  - https://github.com/pioarduino/platform-espressif32/releases | ||||
| PLATFORM_VERSION_LOOKUP = { | ||||
|     "recommended": cv.Version(54, 3, 21, "2"), | ||||
|     "latest": cv.Version(55, 3, 31), | ||||
|     "dev": "https://github.com/pioarduino/platform-espressif32.git#develop", | ||||
| } | ||||
|  | ||||
|  | ||||
| def _check_versions(value): | ||||
|     value = value.copy() | ||||
|     if value[CONF_TYPE] == FRAMEWORK_ARDUINO: | ||||
|         lookups = { | ||||
|             "dev": ( | ||||
|                 cv.Version(3, 2, 1), | ||||
|                 "https://github.com/espressif/arduino-esp32.git", | ||||
|             ), | ||||
|             "latest": (cv.Version(3, 2, 1), None), | ||||
|             "recommended": (RECOMMENDED_ARDUINO_FRAMEWORK_VERSION, None), | ||||
|         } | ||||
|  | ||||
|         if value[CONF_VERSION] in lookups: | ||||
|             if CONF_SOURCE in value: | ||||
|     if value[CONF_VERSION] in PLATFORM_VERSION_LOOKUP: | ||||
|         if CONF_SOURCE in value or CONF_PLATFORM_VERSION in value: | ||||
|             raise cv.Invalid( | ||||
|                     "Framework version needs to be explicitly specified when custom source is used." | ||||
|                 "Version needs to be explicitly set when a custom source or platform_version is used." | ||||
|             ) | ||||
|  | ||||
|             version, source = lookups[value[CONF_VERSION]] | ||||
|         platform_lookup = PLATFORM_VERSION_LOOKUP[value[CONF_VERSION]] | ||||
|         value[CONF_PLATFORM_VERSION] = _parse_platform_version(str(platform_lookup)) | ||||
|  | ||||
|         if value[CONF_TYPE] == FRAMEWORK_ARDUINO: | ||||
|             version = ARDUINO_FRAMEWORK_VERSION_LOOKUP[value[CONF_VERSION]] | ||||
|         else: | ||||
|             version = ESP_IDF_FRAMEWORK_VERSION_LOOKUP[value[CONF_VERSION]] | ||||
|     else: | ||||
|         version = cv.Version.parse(cv.version_number(value[CONF_VERSION])) | ||||
|             source = value.get(CONF_SOURCE, None) | ||||
|  | ||||
|     value[CONF_VERSION] = str(version) | ||||
|         value[CONF_SOURCE] = source or _format_framework_arduino_version(version) | ||||
|  | ||||
|         value[CONF_PLATFORM_VERSION] = value.get( | ||||
|             CONF_PLATFORM_VERSION, | ||||
|             _parse_platform_version(str(ARDUINO_PLATFORM_VERSION)), | ||||
|     if value[CONF_TYPE] == FRAMEWORK_ARDUINO: | ||||
|         if version < cv.Version(3, 0, 0): | ||||
|             raise cv.Invalid("Only Arduino 3.0+ is supported.") | ||||
|         recommended_version = ARDUINO_FRAMEWORK_VERSION_LOOKUP["recommended"] | ||||
|         platform_lookup = ARDUINO_PLATFORM_VERSION_LOOKUP.get(version) | ||||
|         value[CONF_SOURCE] = value.get( | ||||
|             CONF_SOURCE, _format_framework_arduino_version(version) | ||||
|         ) | ||||
|     else: | ||||
|         if version < cv.Version(5, 0, 0): | ||||
|             raise cv.Invalid("Only ESP-IDF 5.0+ is supported.") | ||||
|         recommended_version = ESP_IDF_FRAMEWORK_VERSION_LOOKUP["recommended"] | ||||
|         platform_lookup = ESP_IDF_PLATFORM_VERSION_LOOKUP.get(version) | ||||
|         value[CONF_SOURCE] = value.get( | ||||
|             CONF_SOURCE, | ||||
|             _format_framework_espidf_version(version, value.get(CONF_RELEASE, None)), | ||||
|         ) | ||||
|  | ||||
|         if value[CONF_SOURCE].startswith("http"): | ||||
|             # prefix is necessary or platformio will complain with a cryptic error | ||||
|             value[CONF_SOURCE] = f"framework-arduinoespressif32@{value[CONF_SOURCE]}" | ||||
|     if CONF_PLATFORM_VERSION not in value: | ||||
|         if platform_lookup is None: | ||||
|             raise cv.Invalid( | ||||
|                 "Framework version not recognized; please specify platform_version" | ||||
|             ) | ||||
|         value[CONF_PLATFORM_VERSION] = _parse_platform_version(str(platform_lookup)) | ||||
|  | ||||
|         if version != RECOMMENDED_ARDUINO_FRAMEWORK_VERSION: | ||||
|     if version != recommended_version: | ||||
|         _LOGGER.warning( | ||||
|                 "The selected Arduino framework version is not the recommended one. " | ||||
|             "The selected framework version is not the recommended one. " | ||||
|             "If there are connectivity or build issues please remove the manual version." | ||||
|         ) | ||||
|  | ||||
|         return value | ||||
|  | ||||
|     lookups = { | ||||
|         "dev": (cv.Version(5, 4, 2), "https://github.com/espressif/esp-idf.git"), | ||||
|         "latest": (cv.Version(5, 2, 2), None), | ||||
|         "recommended": (RECOMMENDED_ESP_IDF_FRAMEWORK_VERSION, None), | ||||
|     } | ||||
|  | ||||
|     if value[CONF_VERSION] in lookups: | ||||
|         if CONF_SOURCE in value: | ||||
|             raise cv.Invalid( | ||||
|                 "Framework version needs to be explicitly specified when custom source is used." | ||||
|             ) | ||||
|  | ||||
|         version, source = lookups[value[CONF_VERSION]] | ||||
|     else: | ||||
|         version = cv.Version.parse(cv.version_number(value[CONF_VERSION])) | ||||
|         source = value.get(CONF_SOURCE, None) | ||||
|  | ||||
|     if version < cv.Version(5, 0, 0): | ||||
|         raise cv.Invalid("Only ESP-IDF 5.0+ is supported.") | ||||
|  | ||||
|     # flag this for later *before* we set value[CONF_PLATFORM_VERSION] below | ||||
|     has_platform_ver = CONF_PLATFORM_VERSION in value | ||||
|  | ||||
|     value[CONF_PLATFORM_VERSION] = value.get( | ||||
|         CONF_PLATFORM_VERSION, _parse_platform_version(str(ESP_IDF_PLATFORM_VERSION)) | ||||
|     ) | ||||
|  | ||||
|     if ( | ||||
|         is_platformio := _platform_is_platformio(value[CONF_PLATFORM_VERSION]) | ||||
|     ) and version not in SUPPORTED_PLATFORMIO_ESP_IDF_5X: | ||||
|         raise cv.Invalid( | ||||
|             f"ESP-IDF {str(version)} not supported by platformio/espressif32" | ||||
|         ) | ||||
|  | ||||
|     if ( | ||||
|         version in SUPPORTED_PLATFORMIO_ESP_IDF_5X | ||||
|         and version not in SUPPORTED_PIOARDUINO_ESP_IDF_5X | ||||
|     ) and not has_platform_ver: | ||||
|         raise cv.Invalid( | ||||
|             f"ESP-IDF {value[CONF_VERSION]} may be supported by platformio/espressif32; please specify '{CONF_PLATFORM_VERSION}'" | ||||
|         ) | ||||
|  | ||||
|     if ( | ||||
|         not is_platformio | ||||
|         and CONF_RELEASE not in value | ||||
|         and version not in SUPPORTED_PIOARDUINO_ESP_IDF_5X | ||||
|     if value[CONF_PLATFORM_VERSION] != _parse_platform_version( | ||||
|         str(PLATFORM_VERSION_LOOKUP["recommended"]) | ||||
|     ): | ||||
|         raise cv.Invalid( | ||||
|             f"ESP-IDF {value[CONF_VERSION]} is not available with pioarduino; you may need to specify '{CONF_RELEASE}'" | ||||
|         ) | ||||
|  | ||||
|     value[CONF_VERSION] = str(version) | ||||
|     value[CONF_SOURCE] = source or _format_framework_espidf_version( | ||||
|         version, value.get(CONF_RELEASE, None), is_platformio | ||||
|     ) | ||||
|  | ||||
|     if value[CONF_SOURCE].startswith("http"): | ||||
|         # prefix is necessary or platformio will complain with a cryptic error | ||||
|         value[CONF_SOURCE] = f"framework-espidf@{value[CONF_SOURCE]}" | ||||
|  | ||||
|     if version != RECOMMENDED_ESP_IDF_FRAMEWORK_VERSION: | ||||
|         _LOGGER.warning( | ||||
|             "The selected ESP-IDF framework version is not the recommended one. " | ||||
|             "The selected platform version is not the recommended one. " | ||||
|             "If there are connectivity or build issues please remove the manual version." | ||||
|         ) | ||||
|  | ||||
| @@ -477,26 +423,14 @@ def _check_versions(value): | ||||
| def _parse_platform_version(value): | ||||
|     try: | ||||
|         ver = cv.Version.parse(cv.version_number(value)) | ||||
|         if ver.major >= 50:  # a pioarduino version | ||||
|         release = f"{ver.major}.{ver.minor:02d}.{ver.patch:02d}" | ||||
|         if ver.extra: | ||||
|             release += f"-{ver.extra}" | ||||
|         return f"https://github.com/pioarduino/platform-espressif32/releases/download/{release}/platform-espressif32.zip" | ||||
|         # if platform version is a valid version constraint, prefix the default package | ||||
|         cv.platformio_version_constraint(value) | ||||
|         return f"platformio/espressif32@{value}" | ||||
|     except cv.Invalid: | ||||
|         return value | ||||
|  | ||||
|  | ||||
| def _platform_is_platformio(value): | ||||
|     try: | ||||
|         ver = cv.Version.parse(cv.version_number(value)) | ||||
|         return ver.major < 50 | ||||
|     except cv.Invalid: | ||||
|         return "platformio" in value | ||||
|  | ||||
|  | ||||
| def _detect_variant(value): | ||||
|     board = value.get(CONF_BOARD) | ||||
|     variant = value.get(CONF_VARIANT) | ||||
| @@ -808,6 +742,8 @@ async def to_code(config): | ||||
|  | ||||
|     conf = config[CONF_FRAMEWORK] | ||||
|     cg.add_platformio_option("platform", conf[CONF_PLATFORM_VERSION]) | ||||
|     if CONF_SOURCE in conf: | ||||
|         cg.add_platformio_option("platform_packages", [conf[CONF_SOURCE]]) | ||||
|  | ||||
|     if conf[CONF_ADVANCED][CONF_IGNORE_EFUSE_CUSTOM_MAC]: | ||||
|         cg.add_define("USE_ESP32_IGNORE_EFUSE_CUSTOM_MAC") | ||||
| @@ -850,8 +786,6 @@ async def to_code(config): | ||||
|  | ||||
|     cg.add_build_flag("-Wno-nonnull-compare") | ||||
|  | ||||
|     cg.add_platformio_option("platform_packages", [conf[CONF_SOURCE]]) | ||||
|  | ||||
|     add_idf_sdkconfig_option(f"CONFIG_IDF_TARGET_{variant}", True) | ||||
|     add_idf_sdkconfig_option( | ||||
|         f"CONFIG_ESPTOOLPY_FLASHSIZE_{config[CONF_FLASH_SIZE]}", True | ||||
|   | ||||
| @@ -1,5 +1,8 @@ | ||||
| from collections.abc import Callable, MutableMapping | ||||
| from enum import Enum | ||||
| import logging | ||||
| import re | ||||
| from typing import Any | ||||
|  | ||||
| from esphome import automation | ||||
| import esphome.codegen as cg | ||||
| @@ -9,16 +12,19 @@ from esphome.const import ( | ||||
|     CONF_ENABLE_ON_BOOT, | ||||
|     CONF_ESPHOME, | ||||
|     CONF_ID, | ||||
|     CONF_MAX_CONNECTIONS, | ||||
|     CONF_NAME, | ||||
|     CONF_NAME_ADD_MAC_SUFFIX, | ||||
| ) | ||||
| from esphome.core import TimePeriod | ||||
| from esphome.core import CORE, TimePeriod | ||||
| import esphome.final_validate as fv | ||||
|  | ||||
| DEPENDENCIES = ["esp32"] | ||||
| CODEOWNERS = ["@jesserockz", "@Rapsssito", "@bdraco"] | ||||
| DOMAIN = "esp32_ble" | ||||
|  | ||||
| _LOGGER = logging.getLogger(__name__) | ||||
|  | ||||
|  | ||||
| class BTLoggers(Enum): | ||||
|     """Bluetooth logger categories available in ESP-IDF. | ||||
| @@ -127,6 +133,28 @@ CONF_DISABLE_BT_LOGS = "disable_bt_logs" | ||||
| CONF_CONNECTION_TIMEOUT = "connection_timeout" | ||||
| CONF_MAX_NOTIFICATIONS = "max_notifications" | ||||
|  | ||||
| # BLE connection limits | ||||
| # ESP-IDF CONFIG_BT_ACL_CONNECTIONS has range 1-9, default 4 | ||||
| # Total instances: 10 (ADV + SCAN + connections) | ||||
| # - ADV only: up to 9 connections | ||||
| # - SCAN only: up to 9 connections | ||||
| # - ADV + SCAN: up to 8 connections | ||||
| DEFAULT_MAX_CONNECTIONS = 3 | ||||
| IDF_MAX_CONNECTIONS = 9 | ||||
|  | ||||
| # Connection slot tracking keys | ||||
| KEY_ESP32_BLE = "esp32_ble" | ||||
| KEY_USED_CONNECTION_SLOTS = "used_connection_slots" | ||||
|  | ||||
| # Export for use by other components (bluetooth_proxy, etc.) | ||||
| __all__ = [ | ||||
|     "DEFAULT_MAX_CONNECTIONS", | ||||
|     "IDF_MAX_CONNECTIONS", | ||||
|     "KEY_ESP32_BLE", | ||||
|     "KEY_USED_CONNECTION_SLOTS", | ||||
|     "consume_connection_slots", | ||||
| ] | ||||
|  | ||||
| NO_BLUETOOTH_VARIANTS = [const.VARIANT_ESP32S2] | ||||
|  | ||||
| esp32_ble_ns = cg.esphome_ns.namespace("esp32_ble") | ||||
| @@ -183,6 +211,9 @@ CONFIG_SCHEMA = cv.Schema( | ||||
|             cv.positive_int, | ||||
|             cv.Range(min=1, max=64), | ||||
|         ), | ||||
|         cv.Optional(CONF_MAX_CONNECTIONS, default=DEFAULT_MAX_CONNECTIONS): cv.All( | ||||
|             cv.positive_int, cv.Range(min=1, max=IDF_MAX_CONNECTIONS) | ||||
|         ), | ||||
|     } | ||||
| ).extend(cv.COMPONENT_SCHEMA) | ||||
|  | ||||
| @@ -230,6 +261,56 @@ def validate_variant(_): | ||||
|         raise cv.Invalid(f"{variant} does not support Bluetooth") | ||||
|  | ||||
|  | ||||
| def consume_connection_slots( | ||||
|     value: int, consumer: str | ||||
| ) -> Callable[[MutableMapping], MutableMapping]: | ||||
|     """Reserve BLE connection slots for a component. | ||||
|  | ||||
|     Args: | ||||
|         value: Number of connection slots to reserve | ||||
|         consumer: Name of the component consuming the slots | ||||
|  | ||||
|     Returns: | ||||
|         A validator function that records the slot usage | ||||
|     """ | ||||
|  | ||||
|     def _consume_connection_slots(config: MutableMapping) -> MutableMapping: | ||||
|         data: dict[str, Any] = CORE.data.setdefault(KEY_ESP32_BLE, {}) | ||||
|         slots: list[str] = data.setdefault(KEY_USED_CONNECTION_SLOTS, []) | ||||
|         slots.extend([consumer] * value) | ||||
|         return config | ||||
|  | ||||
|     return _consume_connection_slots | ||||
|  | ||||
|  | ||||
| def validate_connection_slots(max_connections: int) -> None: | ||||
|     """Validate that BLE connection slots don't exceed the configured maximum.""" | ||||
|     ble_data = CORE.data.get(KEY_ESP32_BLE, {}) | ||||
|     used_slots = ble_data.get(KEY_USED_CONNECTION_SLOTS, []) | ||||
|     num_used = len(used_slots) | ||||
|  | ||||
|     if num_used <= max_connections: | ||||
|         return | ||||
|  | ||||
|     slot_users = ", ".join(used_slots) | ||||
|  | ||||
|     if num_used > IDF_MAX_CONNECTIONS: | ||||
|         raise cv.Invalid( | ||||
|             f"BLE components require {num_used} connection slots but maximum is {IDF_MAX_CONNECTIONS}. " | ||||
|             f"Reduce the number of BLE clients. Components: {slot_users}" | ||||
|         ) | ||||
|  | ||||
|     _LOGGER.warning( | ||||
|         "BLE components require %d connection slot(s) but only %d configured. " | ||||
|         "Please set 'max_connections: %d' in the 'esp32_ble' component. " | ||||
|         "Components: %s", | ||||
|         num_used, | ||||
|         max_connections, | ||||
|         num_used, | ||||
|         slot_users, | ||||
|     ) | ||||
|  | ||||
|  | ||||
| def final_validation(config): | ||||
|     validate_variant(config) | ||||
|     if (name := config.get(CONF_NAME)) is not None: | ||||
| @@ -245,6 +326,10 @@ def final_validation(config): | ||||
|     # Set GATT Client/Server sdkconfig options based on which components are loaded | ||||
|     full_config = fv.full_config.get() | ||||
|  | ||||
|     # Validate connection slots usage | ||||
|     max_connections = config.get(CONF_MAX_CONNECTIONS, DEFAULT_MAX_CONNECTIONS) | ||||
|     validate_connection_slots(max_connections) | ||||
|  | ||||
|     # Check if BLE Server is needed | ||||
|     has_ble_server = "esp32_ble_server" in full_config | ||||
|     add_idf_sdkconfig_option("CONFIG_BT_GATTS_ENABLE", has_ble_server) | ||||
| @@ -255,6 +340,26 @@ def final_validation(config): | ||||
|     ) | ||||
|     add_idf_sdkconfig_option("CONFIG_BT_GATTC_ENABLE", has_ble_client) | ||||
|  | ||||
|     # Handle max_connections: check for deprecated location in esp32_ble_tracker | ||||
|     max_connections = config.get(CONF_MAX_CONNECTIONS, DEFAULT_MAX_CONNECTIONS) | ||||
|  | ||||
|     # Use value from tracker if esp32_ble doesn't have it explicitly set (backward compat) | ||||
|     if "esp32_ble_tracker" in full_config: | ||||
|         tracker_config = full_config["esp32_ble_tracker"] | ||||
|         if "max_connections" in tracker_config and CONF_MAX_CONNECTIONS not in config: | ||||
|             max_connections = tracker_config["max_connections"] | ||||
|  | ||||
|     # Set CONFIG_BT_ACL_CONNECTIONS to the maximum connections needed + 1 for ADV/SCAN | ||||
|     # This is the Bluedroid host stack total instance limit (range 1-9, default 4) | ||||
|     # Total instances = ADV/SCAN (1) + connection slots (max_connections) | ||||
|     # Shared between client (tracker/ble_client) and server | ||||
|     add_idf_sdkconfig_option("CONFIG_BT_ACL_CONNECTIONS", max_connections + 1) | ||||
|  | ||||
|     # Set controller-specific max connections for ESP32 (classic) | ||||
|     # CONFIG_BTDM_CTRL_BLE_MAX_CONN is ESP32-specific controller limit (just connections, not ADV/SCAN) | ||||
|     # For newer chips (C3/S3/etc), different configs are used automatically | ||||
|     add_idf_sdkconfig_option("CONFIG_BTDM_CTRL_BLE_MAX_CONN", max_connections) | ||||
|  | ||||
|     return config | ||||
|  | ||||
|  | ||||
| @@ -270,6 +375,10 @@ async def to_code(config): | ||||
|         cg.add(var.set_name(name)) | ||||
|     await cg.register_component(var, config) | ||||
|  | ||||
|     # Define max connections for use in C++ code (e.g., ble_server.h) | ||||
|     max_connections = config.get(CONF_MAX_CONNECTIONS, DEFAULT_MAX_CONNECTIONS) | ||||
|     cg.add_define("USE_ESP32_BLE_MAX_CONNECTIONS", max_connections) | ||||
|  | ||||
|     add_idf_sdkconfig_option("CONFIG_BT_ENABLED", True) | ||||
|     add_idf_sdkconfig_option("CONFIG_BT_BLE_42_FEATURES_SUPPORTED", True) | ||||
|  | ||||
|   | ||||
| @@ -213,15 +213,17 @@ bool ESP32BLE::ble_setup_() { | ||||
|   if (this->name_.has_value()) { | ||||
|     name = this->name_.value(); | ||||
|     if (App.is_name_add_mac_suffix_enabled()) { | ||||
|       name += "-" + get_mac_address().substr(6); | ||||
|       name += "-"; | ||||
|       name += get_mac_address().substr(6); | ||||
|     } | ||||
|   } else { | ||||
|     name = App.get_name(); | ||||
|     if (name.length() > 20) { | ||||
|       if (App.is_name_add_mac_suffix_enabled()) { | ||||
|         name.erase(name.begin() + 13, name.end() - 7);  // Remove characters between 13 and the mac address | ||||
|         // Keep first 13 chars and last 7 chars (MAC suffix), remove middle | ||||
|         name.erase(13, name.length() - 20); | ||||
|       } else { | ||||
|         name = name.substr(0, 20); | ||||
|         name.resize(20); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   | ||||
| @@ -152,7 +152,7 @@ void BLEAdvertising::loop() { | ||||
|   if (now - this->last_advertisement_time_ > this->advertising_cycle_time_) { | ||||
|     this->stop(); | ||||
|     this->current_adv_index_ += 1; | ||||
|     if (this->current_adv_index_ >= this->raw_advertisements_callbacks_.size()) { | ||||
|     if (static_cast<size_t>(this->current_adv_index_) >= this->raw_advertisements_callbacks_.size()) { | ||||
|       this->current_adv_index_ = -1; | ||||
|     } | ||||
|     this->start(); | ||||
|   | ||||
| @@ -26,7 +26,7 @@ from esphome.const import ( | ||||
| from esphome.core import CORE | ||||
| from esphome.schema_extractors import SCHEMA_EXTRACT | ||||
|  | ||||
| AUTO_LOAD = ["esp32_ble", "bytebuffer", "event_emitter"] | ||||
| AUTO_LOAD = ["esp32_ble", "bytebuffer"] | ||||
| CODEOWNERS = ["@jesserockz", "@clydebarrow", "@Rapsssito"] | ||||
| DEPENDENCIES = ["esp32"] | ||||
| DOMAIN = "esp32_ble_server" | ||||
|   | ||||
| @@ -49,7 +49,11 @@ void BLECharacteristic::notify() { | ||||
|       this->service_->get_server()->get_connected_client_count() == 0) | ||||
|     return; | ||||
|  | ||||
|   for (auto &client : this->service_->get_server()->get_clients()) { | ||||
|   const uint16_t *clients = this->service_->get_server()->get_clients(); | ||||
|   uint8_t client_count = this->service_->get_server()->get_client_count(); | ||||
|  | ||||
|   for (uint8_t i = 0; i < client_count; i++) { | ||||
|     uint16_t client = clients[i]; | ||||
|     size_t length = this->value_.size(); | ||||
|     // Find the client in the list of clients to notify | ||||
|     auto *entry = this->find_client_in_notify_list_(client); | ||||
| @@ -73,7 +77,7 @@ void BLECharacteristic::notify() { | ||||
| void BLECharacteristic::add_descriptor(BLEDescriptor *descriptor) { | ||||
|   // If the descriptor is the CCCD descriptor, listen to its write event to know if the client wants to be notified | ||||
|   if (descriptor->get_uuid() == ESPBTUUID::from_uint16(ESP_GATT_UUID_CHAR_CLIENT_CONFIG)) { | ||||
|     descriptor->on(BLEDescriptorEvt::VectorEvt::ON_WRITE, [this](const std::vector<uint8_t> &value, uint16_t conn_id) { | ||||
|     descriptor->on_write([this](std::span<const uint8_t> value, uint16_t conn_id) { | ||||
|       if (value.size() != 2) | ||||
|         return; | ||||
|       uint16_t cccd = encode_uint16(value[1], value[0]); | ||||
| @@ -121,69 +125,49 @@ bool BLECharacteristic::is_created() { | ||||
|   if (this->state_ != CREATING_DEPENDENTS) | ||||
|     return false; | ||||
|  | ||||
|   bool created = true; | ||||
|   for (auto *descriptor : this->descriptors_) { | ||||
|     created &= descriptor->is_created(); | ||||
|     if (!descriptor->is_created()) | ||||
|       return false; | ||||
|   } | ||||
|   if (created) | ||||
|   // All descriptors are created if we reach here | ||||
|   this->state_ = CREATED; | ||||
|   return this->state_ == CREATED; | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| bool BLECharacteristic::is_failed() { | ||||
|   if (this->state_ == FAILED) | ||||
|     return true; | ||||
|  | ||||
|   bool failed = false; | ||||
|   for (auto *descriptor : this->descriptors_) { | ||||
|     failed |= descriptor->is_failed(); | ||||
|   } | ||||
|   if (failed) | ||||
|     if (descriptor->is_failed()) { | ||||
|       this->state_ = FAILED; | ||||
|   return this->state_ == FAILED; | ||||
|       return true; | ||||
|     } | ||||
|   } | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| void BLECharacteristic::set_property_bit_(esp_gatt_char_prop_t bit, bool value) { | ||||
|   if (value) { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ | bit); | ||||
|   } else { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ & ~bit); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void BLECharacteristic::set_broadcast_property(bool value) { | ||||
|   if (value) { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ | ESP_GATT_CHAR_PROP_BIT_BROADCAST); | ||||
|   } else { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ & ~ESP_GATT_CHAR_PROP_BIT_BROADCAST); | ||||
|   } | ||||
|   this->set_property_bit_(ESP_GATT_CHAR_PROP_BIT_BROADCAST, value); | ||||
| } | ||||
| void BLECharacteristic::set_indicate_property(bool value) { | ||||
|   if (value) { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ | ESP_GATT_CHAR_PROP_BIT_INDICATE); | ||||
|   } else { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ & ~ESP_GATT_CHAR_PROP_BIT_INDICATE); | ||||
|   } | ||||
|   this->set_property_bit_(ESP_GATT_CHAR_PROP_BIT_INDICATE, value); | ||||
| } | ||||
| void BLECharacteristic::set_notify_property(bool value) { | ||||
|   if (value) { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ | ESP_GATT_CHAR_PROP_BIT_NOTIFY); | ||||
|   } else { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ & ~ESP_GATT_CHAR_PROP_BIT_NOTIFY); | ||||
|   } | ||||
| } | ||||
| void BLECharacteristic::set_read_property(bool value) { | ||||
|   if (value) { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ | ESP_GATT_CHAR_PROP_BIT_READ); | ||||
|   } else { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ & ~ESP_GATT_CHAR_PROP_BIT_READ); | ||||
|   } | ||||
| } | ||||
| void BLECharacteristic::set_write_property(bool value) { | ||||
|   if (value) { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ | ESP_GATT_CHAR_PROP_BIT_WRITE); | ||||
|   } else { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ & ~ESP_GATT_CHAR_PROP_BIT_WRITE); | ||||
|   } | ||||
|   this->set_property_bit_(ESP_GATT_CHAR_PROP_BIT_NOTIFY, value); | ||||
| } | ||||
| void BLECharacteristic::set_read_property(bool value) { this->set_property_bit_(ESP_GATT_CHAR_PROP_BIT_READ, value); } | ||||
| void BLECharacteristic::set_write_property(bool value) { this->set_property_bit_(ESP_GATT_CHAR_PROP_BIT_WRITE, value); } | ||||
| void BLECharacteristic::set_write_no_response_property(bool value) { | ||||
|   if (value) { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ | ESP_GATT_CHAR_PROP_BIT_WRITE_NR); | ||||
|   } else { | ||||
|     this->properties_ = (esp_gatt_char_prop_t) (this->properties_ & ~ESP_GATT_CHAR_PROP_BIT_WRITE_NR); | ||||
|   } | ||||
|   this->set_property_bit_(ESP_GATT_CHAR_PROP_BIT_WRITE_NR, value); | ||||
| } | ||||
|  | ||||
| void BLECharacteristic::gatts_event_handler(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, | ||||
| @@ -208,8 +192,9 @@ void BLECharacteristic::gatts_event_handler(esp_gatts_cb_event_t event, esp_gatt | ||||
|       if (!param->read.need_rsp) | ||||
|         break;  // For some reason you can request a read but not want a response | ||||
|  | ||||
|       this->EventEmitter<BLECharacteristicEvt::EmptyEvt, uint16_t>::emit_(BLECharacteristicEvt::EmptyEvt::ON_READ, | ||||
|                                                                           param->read.conn_id); | ||||
|       if (this->on_read_callback_) { | ||||
|         (*this->on_read_callback_)(param->read.conn_id); | ||||
|       } | ||||
|  | ||||
|       uint16_t max_offset = 22; | ||||
|  | ||||
| @@ -277,8 +262,9 @@ void BLECharacteristic::gatts_event_handler(esp_gatts_cb_event_t event, esp_gatt | ||||
|       } | ||||
|  | ||||
|       if (!param->write.is_prep) { | ||||
|         this->EventEmitter<BLECharacteristicEvt::VectorEvt, std::vector<uint8_t>, uint16_t>::emit_( | ||||
|             BLECharacteristicEvt::VectorEvt::ON_WRITE, this->value_, param->write.conn_id); | ||||
|         if (this->on_write_callback_) { | ||||
|           (*this->on_write_callback_)(this->value_, param->write.conn_id); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       break; | ||||
| @@ -289,8 +275,9 @@ void BLECharacteristic::gatts_event_handler(esp_gatts_cb_event_t event, esp_gatt | ||||
|         break; | ||||
|       this->write_event_ = false; | ||||
|       if (param->exec_write.exec_write_flag == ESP_GATT_PREP_WRITE_EXEC) { | ||||
|         this->EventEmitter<BLECharacteristicEvt::VectorEvt, std::vector<uint8_t>, uint16_t>::emit_( | ||||
|             BLECharacteristicEvt::VectorEvt::ON_WRITE, this->value_, param->exec_write.conn_id); | ||||
|         if (this->on_write_callback_) { | ||||
|           (*this->on_write_callback_)(this->value_, param->exec_write.conn_id); | ||||
|         } | ||||
|       } | ||||
|       esp_err_t err = | ||||
|           esp_ble_gatts_send_response(gatts_if, param->write.conn_id, param->write.trans_id, ESP_GATT_OK, nullptr); | ||||
|   | ||||
| @@ -2,10 +2,12 @@ | ||||
|  | ||||
| #include "ble_descriptor.h" | ||||
| #include "esphome/components/esp32_ble/ble_uuid.h" | ||||
| #include "esphome/components/event_emitter/event_emitter.h" | ||||
| #include "esphome/components/bytebuffer/bytebuffer.h" | ||||
|  | ||||
| #include <vector> | ||||
| #include <span> | ||||
| #include <functional> | ||||
| #include <memory> | ||||
|  | ||||
| #ifdef USE_ESP32 | ||||
|  | ||||
| @@ -22,22 +24,10 @@ namespace esp32_ble_server { | ||||
|  | ||||
| using namespace esp32_ble; | ||||
| using namespace bytebuffer; | ||||
| using namespace event_emitter; | ||||
|  | ||||
| class BLEService; | ||||
|  | ||||
| namespace BLECharacteristicEvt { | ||||
| enum VectorEvt { | ||||
|   ON_WRITE, | ||||
| }; | ||||
|  | ||||
| enum EmptyEvt { | ||||
|   ON_READ, | ||||
| }; | ||||
| }  // namespace BLECharacteristicEvt | ||||
|  | ||||
| class BLECharacteristic : public EventEmitter<BLECharacteristicEvt::VectorEvt, std::vector<uint8_t>, uint16_t>, | ||||
|                           public EventEmitter<BLECharacteristicEvt::EmptyEvt, uint16_t> { | ||||
| class BLECharacteristic { | ||||
|  public: | ||||
|   BLECharacteristic(ESPBTUUID uuid, uint32_t properties); | ||||
|   ~BLECharacteristic(); | ||||
| @@ -76,6 +66,15 @@ class BLECharacteristic : public EventEmitter<BLECharacteristicEvt::VectorEvt, s | ||||
|   bool is_created(); | ||||
|   bool is_failed(); | ||||
|  | ||||
|   // Direct callback registration - only allocates when callback is set | ||||
|   void on_write(std::function<void(std::span<const uint8_t>, uint16_t)> &&callback) { | ||||
|     this->on_write_callback_ = | ||||
|         std::make_unique<std::function<void(std::span<const uint8_t>, uint16_t)>>(std::move(callback)); | ||||
|   } | ||||
|   void on_read(std::function<void(uint16_t)> &&callback) { | ||||
|     this->on_read_callback_ = std::make_unique<std::function<void(uint16_t)>>(std::move(callback)); | ||||
|   } | ||||
|  | ||||
|  protected: | ||||
|   bool write_event_{false}; | ||||
|   BLEService *service_{}; | ||||
| @@ -98,6 +97,11 @@ class BLECharacteristic : public EventEmitter<BLECharacteristicEvt::VectorEvt, s | ||||
|   void remove_client_from_notify_list_(uint16_t conn_id); | ||||
|   ClientNotificationEntry *find_client_in_notify_list_(uint16_t conn_id); | ||||
|  | ||||
|   void set_property_bit_(esp_gatt_char_prop_t bit, bool value); | ||||
|  | ||||
|   std::unique_ptr<std::function<void(std::span<const uint8_t>, uint16_t)>> on_write_callback_; | ||||
|   std::unique_ptr<std::function<void(uint16_t)>> on_read_callback_; | ||||
|  | ||||
|   esp_gatt_perm_t permissions_ = ESP_GATT_PERM_READ | ESP_GATT_PERM_WRITE; | ||||
|  | ||||
|   enum State : uint8_t { | ||||
|   | ||||
| @@ -74,9 +74,10 @@ void BLEDescriptor::gatts_event_handler(esp_gatts_cb_event_t event, esp_gatt_if_ | ||||
|         break; | ||||
|       this->value_.attr_len = param->write.len; | ||||
|       memcpy(this->value_.attr_value, param->write.value, param->write.len); | ||||
|       this->emit_(BLEDescriptorEvt::VectorEvt::ON_WRITE, | ||||
|                   std::vector<uint8_t>(param->write.value, param->write.value + param->write.len), | ||||
|       if (this->on_write_callback_) { | ||||
|         (*this->on_write_callback_)(std::span<const uint8_t>(param->write.value, param->write.len), | ||||
|                                     param->write.conn_id); | ||||
|       } | ||||
|       break; | ||||
|     } | ||||
|     default: | ||||
|   | ||||
| @@ -1,30 +1,26 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "esphome/components/esp32_ble/ble_uuid.h" | ||||
| #include "esphome/components/event_emitter/event_emitter.h" | ||||
| #include "esphome/components/bytebuffer/bytebuffer.h" | ||||
|  | ||||
| #ifdef USE_ESP32 | ||||
|  | ||||
| #include <esp_gatt_defs.h> | ||||
| #include <esp_gatts_api.h> | ||||
| #include <span> | ||||
| #include <functional> | ||||
| #include <memory> | ||||
|  | ||||
| namespace esphome { | ||||
| namespace esp32_ble_server { | ||||
|  | ||||
| using namespace esp32_ble; | ||||
| using namespace bytebuffer; | ||||
| using namespace event_emitter; | ||||
|  | ||||
| class BLECharacteristic; | ||||
|  | ||||
| namespace BLEDescriptorEvt { | ||||
| enum VectorEvt { | ||||
|   ON_WRITE, | ||||
| }; | ||||
| }  // namespace BLEDescriptorEvt | ||||
|  | ||||
| class BLEDescriptor : public EventEmitter<BLEDescriptorEvt::VectorEvt, std::vector<uint8_t>, uint16_t> { | ||||
| // Base class for BLE descriptors | ||||
| class BLEDescriptor { | ||||
|  public: | ||||
|   BLEDescriptor(ESPBTUUID uuid, uint16_t max_len = 100, bool read = true, bool write = true); | ||||
|   virtual ~BLEDescriptor(); | ||||
| @@ -39,6 +35,12 @@ class BLEDescriptor : public EventEmitter<BLEDescriptorEvt::VectorEvt, std::vect | ||||
|   bool is_created() { return this->state_ == CREATED; } | ||||
|   bool is_failed() { return this->state_ == FAILED; } | ||||
|  | ||||
|   // Direct callback registration - only allocates when callback is set | ||||
|   void on_write(std::function<void(std::span<const uint8_t>, uint16_t)> &&callback) { | ||||
|     this->on_write_callback_ = | ||||
|         std::make_unique<std::function<void(std::span<const uint8_t>, uint16_t)>>(std::move(callback)); | ||||
|   } | ||||
|  | ||||
|  protected: | ||||
|   BLECharacteristic *characteristic_{nullptr}; | ||||
|   ESPBTUUID uuid_; | ||||
| @@ -46,6 +48,8 @@ class BLEDescriptor : public EventEmitter<BLEDescriptorEvt::VectorEvt, std::vect | ||||
|  | ||||
|   esp_attr_value_t value_{}; | ||||
|  | ||||
|   std::unique_ptr<std::function<void(std::span<const uint8_t>, uint16_t)>> on_write_callback_; | ||||
|  | ||||
|   esp_gatt_perm_t permissions_{}; | ||||
|  | ||||
|   enum State : uint8_t { | ||||
|   | ||||
| @@ -147,20 +147,28 @@ BLEService *BLEServer::get_service(ESPBTUUID uuid, uint8_t inst_id) { | ||||
|   return nullptr; | ||||
| } | ||||
|  | ||||
| void BLEServer::dispatch_callbacks_(CallbackType type, uint16_t conn_id) { | ||||
|   for (auto &entry : this->callbacks_) { | ||||
|     if (entry.type == type) { | ||||
|       entry.callback(conn_id); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| void BLEServer::gatts_event_handler(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, | ||||
|                                     esp_ble_gatts_cb_param_t *param) { | ||||
|   switch (event) { | ||||
|     case ESP_GATTS_CONNECT_EVT: { | ||||
|       ESP_LOGD(TAG, "BLE Client connected"); | ||||
|       this->add_client_(param->connect.conn_id); | ||||
|       this->emit_(BLEServerEvt::EmptyEvt::ON_CONNECT, param->connect.conn_id); | ||||
|       this->dispatch_callbacks_(CallbackType::ON_CONNECT, param->connect.conn_id); | ||||
|       break; | ||||
|     } | ||||
|     case ESP_GATTS_DISCONNECT_EVT: { | ||||
|       ESP_LOGD(TAG, "BLE Client disconnected"); | ||||
|       this->remove_client_(param->disconnect.conn_id); | ||||
|       this->parent_->advertising_start(); | ||||
|       this->emit_(BLEServerEvt::EmptyEvt::ON_DISCONNECT, param->disconnect.conn_id); | ||||
|       this->dispatch_callbacks_(CallbackType::ON_DISCONNECT, param->disconnect.conn_id); | ||||
|       break; | ||||
|     } | ||||
|     case ESP_GATTS_REG_EVT: { | ||||
| @@ -177,9 +185,38 @@ void BLEServer::gatts_event_handler(esp_gatts_cb_event_t event, esp_gatt_if_t ga | ||||
|   } | ||||
| } | ||||
|  | ||||
| int8_t BLEServer::find_client_index_(uint16_t conn_id) const { | ||||
|   for (uint8_t i = 0; i < this->client_count_; i++) { | ||||
|     if (this->clients_[i] == conn_id) | ||||
|       return i; | ||||
|   } | ||||
|   return -1; | ||||
| } | ||||
|  | ||||
| void BLEServer::add_client_(uint16_t conn_id) { | ||||
|   // Check if already in list | ||||
|   if (this->find_client_index_(conn_id) >= 0) | ||||
|     return; | ||||
|   // Add if there's space | ||||
|   if (this->client_count_ < USE_ESP32_BLE_MAX_CONNECTIONS) { | ||||
|     this->clients_[this->client_count_++] = conn_id; | ||||
|   } else { | ||||
|     // This should never happen since max clients is known at compile time | ||||
|     ESP_LOGE(TAG, "Client array full"); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void BLEServer::remove_client_(uint16_t conn_id) { | ||||
|   int8_t index = this->find_client_index_(conn_id); | ||||
|   if (index >= 0) { | ||||
|     // Replace with last element and decrement count (client order not preserved) | ||||
|     this->clients_[index] = this->clients_[--this->client_count_]; | ||||
|   } | ||||
| } | ||||
|  | ||||
| void BLEServer::ble_before_disabled_event_handler() { | ||||
|   // Delete all clients | ||||
|   this->clients_.clear(); | ||||
|   this->client_count_ = 0; | ||||
|   // Delete all services | ||||
|   for (auto &entry : this->services_) { | ||||
|     entry.service->do_delete(); | ||||
|   | ||||
| @@ -12,7 +12,7 @@ | ||||
| #include <memory> | ||||
| #include <vector> | ||||
| #include <unordered_map> | ||||
| #include <unordered_set> | ||||
| #include <functional> | ||||
|  | ||||
| #ifdef USE_ESP32 | ||||
|  | ||||
| @@ -24,18 +24,7 @@ namespace esp32_ble_server { | ||||
| using namespace esp32_ble; | ||||
| using namespace bytebuffer; | ||||
|  | ||||
| namespace BLEServerEvt { | ||||
| enum EmptyEvt { | ||||
|   ON_CONNECT, | ||||
|   ON_DISCONNECT, | ||||
| }; | ||||
| }  // namespace BLEServerEvt | ||||
|  | ||||
| class BLEServer : public Component, | ||||
|                   public GATTsEventHandler, | ||||
|                   public BLEStatusEventHandler, | ||||
|                   public Parented<ESP32BLE>, | ||||
|                   public EventEmitter<BLEServerEvt::EmptyEvt, uint16_t> { | ||||
| class BLEServer : public Component, public GATTsEventHandler, public BLEStatusEventHandler, public Parented<ESP32BLE> { | ||||
|  public: | ||||
|   void setup() override; | ||||
|   void loop() override; | ||||
| @@ -57,15 +46,34 @@ class BLEServer : public Component, | ||||
|   void set_device_information_service(BLEService *service) { this->device_information_service_ = service; } | ||||
|  | ||||
|   esp_gatt_if_t get_gatts_if() { return this->gatts_if_; } | ||||
|   uint32_t get_connected_client_count() { return this->clients_.size(); } | ||||
|   const std::unordered_set<uint16_t> &get_clients() { return this->clients_; } | ||||
|   uint32_t get_connected_client_count() { return this->client_count_; } | ||||
|   const uint16_t *get_clients() const { return this->clients_; } | ||||
|   uint8_t get_client_count() const { return this->client_count_; } | ||||
|  | ||||
|   void gatts_event_handler(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, | ||||
|                            esp_ble_gatts_cb_param_t *param) override; | ||||
|  | ||||
|   void ble_before_disabled_event_handler() override; | ||||
|  | ||||
|   // Direct callback registration - supports multiple callbacks | ||||
|   void on_connect(std::function<void(uint16_t)> &&callback) { | ||||
|     this->callbacks_.push_back({CallbackType::ON_CONNECT, std::move(callback)}); | ||||
|   } | ||||
|   void on_disconnect(std::function<void(uint16_t)> &&callback) { | ||||
|     this->callbacks_.push_back({CallbackType::ON_DISCONNECT, std::move(callback)}); | ||||
|   } | ||||
|  | ||||
|  protected: | ||||
|   enum class CallbackType : uint8_t { | ||||
|     ON_CONNECT, | ||||
|     ON_DISCONNECT, | ||||
|   }; | ||||
|  | ||||
|   struct CallbackEntry { | ||||
|     CallbackType type; | ||||
|     std::function<void(uint16_t)> callback; | ||||
|   }; | ||||
|  | ||||
|   struct ServiceEntry { | ||||
|     ESPBTUUID uuid; | ||||
|     uint8_t inst_id; | ||||
| @@ -74,14 +82,19 @@ class BLEServer : public Component, | ||||
|  | ||||
|   void restart_advertising_(); | ||||
|  | ||||
|   void add_client_(uint16_t conn_id) { this->clients_.insert(conn_id); } | ||||
|   void remove_client_(uint16_t conn_id) { this->clients_.erase(conn_id); } | ||||
|   int8_t find_client_index_(uint16_t conn_id) const; | ||||
|   void add_client_(uint16_t conn_id); | ||||
|   void remove_client_(uint16_t conn_id); | ||||
|   void dispatch_callbacks_(CallbackType type, uint16_t conn_id); | ||||
|  | ||||
|   std::vector<CallbackEntry> callbacks_; | ||||
|  | ||||
|   std::vector<uint8_t> manufacturer_data_{}; | ||||
|   esp_gatt_if_t gatts_if_{0}; | ||||
|   bool registered_{false}; | ||||
|  | ||||
|   std::unordered_set<uint16_t> clients_; | ||||
|   uint16_t clients_[USE_ESP32_BLE_MAX_CONNECTIONS]{}; | ||||
|   uint8_t client_count_{0}; | ||||
|   std::vector<ServiceEntry> services_{}; | ||||
|   std::vector<BLEService *> services_to_start_{}; | ||||
|   BLEService *device_information_service_{}; | ||||
|   | ||||
| @@ -14,9 +14,10 @@ Trigger<std::vector<uint8_t>, uint16_t> *BLETriggers::create_characteristic_on_w | ||||
|     BLECharacteristic *characteristic) { | ||||
|   Trigger<std::vector<uint8_t>, uint16_t> *on_write_trigger =  // NOLINT(cppcoreguidelines-owning-memory) | ||||
|       new Trigger<std::vector<uint8_t>, uint16_t>(); | ||||
|   characteristic->EventEmitter<BLECharacteristicEvt::VectorEvt, std::vector<uint8_t>, uint16_t>::on( | ||||
|       BLECharacteristicEvt::VectorEvt::ON_WRITE, | ||||
|       [on_write_trigger](const std::vector<uint8_t> &data, uint16_t id) { on_write_trigger->trigger(data, id); }); | ||||
|   characteristic->on_write([on_write_trigger](std::span<const uint8_t> data, uint16_t id) { | ||||
|     // Convert span to vector for trigger | ||||
|     on_write_trigger->trigger(std::vector<uint8_t>(data.begin(), data.end()), id); | ||||
|   }); | ||||
|   return on_write_trigger; | ||||
| } | ||||
| #endif | ||||
| @@ -25,9 +26,10 @@ Trigger<std::vector<uint8_t>, uint16_t> *BLETriggers::create_characteristic_on_w | ||||
| Trigger<std::vector<uint8_t>, uint16_t> *BLETriggers::create_descriptor_on_write_trigger(BLEDescriptor *descriptor) { | ||||
|   Trigger<std::vector<uint8_t>, uint16_t> *on_write_trigger =  // NOLINT(cppcoreguidelines-owning-memory) | ||||
|       new Trigger<std::vector<uint8_t>, uint16_t>(); | ||||
|   descriptor->on( | ||||
|       BLEDescriptorEvt::VectorEvt::ON_WRITE, | ||||
|       [on_write_trigger](const std::vector<uint8_t> &data, uint16_t id) { on_write_trigger->trigger(data, id); }); | ||||
|   descriptor->on_write([on_write_trigger](std::span<const uint8_t> data, uint16_t id) { | ||||
|     // Convert span to vector for trigger | ||||
|     on_write_trigger->trigger(std::vector<uint8_t>(data.begin(), data.end()), id); | ||||
|   }); | ||||
|   return on_write_trigger; | ||||
| } | ||||
| #endif | ||||
| @@ -35,8 +37,7 @@ Trigger<std::vector<uint8_t>, uint16_t> *BLETriggers::create_descriptor_on_write | ||||
| #ifdef USE_ESP32_BLE_SERVER_ON_CONNECT | ||||
| Trigger<uint16_t> *BLETriggers::create_server_on_connect_trigger(BLEServer *server) { | ||||
|   Trigger<uint16_t> *on_connect_trigger = new Trigger<uint16_t>();  // NOLINT(cppcoreguidelines-owning-memory) | ||||
|   server->on(BLEServerEvt::EmptyEvt::ON_CONNECT, | ||||
|              [on_connect_trigger](uint16_t conn_id) { on_connect_trigger->trigger(conn_id); }); | ||||
|   server->on_connect([on_connect_trigger](uint16_t conn_id) { on_connect_trigger->trigger(conn_id); }); | ||||
|   return on_connect_trigger; | ||||
| } | ||||
| #endif | ||||
| @@ -44,38 +45,22 @@ Trigger<uint16_t> *BLETriggers::create_server_on_connect_trigger(BLEServer *serv | ||||
| #ifdef USE_ESP32_BLE_SERVER_ON_DISCONNECT | ||||
| Trigger<uint16_t> *BLETriggers::create_server_on_disconnect_trigger(BLEServer *server) { | ||||
|   Trigger<uint16_t> *on_disconnect_trigger = new Trigger<uint16_t>();  // NOLINT(cppcoreguidelines-owning-memory) | ||||
|   server->on(BLEServerEvt::EmptyEvt::ON_DISCONNECT, | ||||
|              [on_disconnect_trigger](uint16_t conn_id) { on_disconnect_trigger->trigger(conn_id); }); | ||||
|   server->on_disconnect([on_disconnect_trigger](uint16_t conn_id) { on_disconnect_trigger->trigger(conn_id); }); | ||||
|   return on_disconnect_trigger; | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_ESP32_BLE_SERVER_SET_VALUE_ACTION | ||||
| void BLECharacteristicSetValueActionManager::set_listener(BLECharacteristic *characteristic, | ||||
|                                                           EventEmitterListenerID listener_id, | ||||
|                                                           const std::function<void()> &pre_notify_listener) { | ||||
|   // Find and remove existing listener for this characteristic | ||||
|   auto *existing = this->find_listener_(characteristic); | ||||
|   if (existing != nullptr) { | ||||
|     // Remove the previous listener | ||||
|     characteristic->EventEmitter<BLECharacteristicEvt::EmptyEvt, uint16_t>::off(BLECharacteristicEvt::EmptyEvt::ON_READ, | ||||
|                                                                                 existing->listener_id); | ||||
|     // Remove the pre-notify listener | ||||
|     this->off(BLECharacteristicSetValueActionEvt::PRE_NOTIFY, existing->pre_notify_listener_id); | ||||
|     // Remove from vector | ||||
|     this->remove_listener_(characteristic); | ||||
|   } | ||||
|   // Create a new listener for the pre-notify event | ||||
|   EventEmitterListenerID pre_notify_listener_id = | ||||
|       this->on(BLECharacteristicSetValueActionEvt::PRE_NOTIFY, | ||||
|                [pre_notify_listener, characteristic](const BLECharacteristic *evt_characteristic) { | ||||
|                  // Only call the pre-notify listener if the characteristic is the one we are interested in | ||||
|                  if (characteristic == evt_characteristic) { | ||||
|                    pre_notify_listener(); | ||||
|                  } | ||||
|                }); | ||||
|   // Save the entry to the vector | ||||
|   this->listeners_.push_back({characteristic, listener_id, pre_notify_listener_id}); | ||||
|   this->listeners_.push_back({characteristic, pre_notify_listener}); | ||||
| } | ||||
|  | ||||
| BLECharacteristicSetValueActionManager::ListenerEntry *BLECharacteristicSetValueActionManager::find_listener_( | ||||
|   | ||||
| @@ -4,7 +4,6 @@ | ||||
| #include "ble_characteristic.h" | ||||
| #include "ble_descriptor.h" | ||||
|  | ||||
| #include "esphome/components/event_emitter/event_emitter.h" | ||||
| #include "esphome/core/automation.h" | ||||
|  | ||||
| #include <vector> | ||||
| @@ -18,10 +17,6 @@ namespace esp32_ble_server { | ||||
| namespace esp32_ble_server_automations { | ||||
|  | ||||
| using namespace esp32_ble; | ||||
| using namespace event_emitter; | ||||
|  | ||||
| // Invalid listener ID constant - 0 is used as sentinel value in EventEmitter | ||||
| static constexpr EventEmitterListenerID INVALID_LISTENER_ID = 0; | ||||
|  | ||||
| class BLETriggers { | ||||
|  public: | ||||
| @@ -41,38 +36,29 @@ class BLETriggers { | ||||
| }; | ||||
|  | ||||
| #ifdef USE_ESP32_BLE_SERVER_SET_VALUE_ACTION | ||||
| enum BLECharacteristicSetValueActionEvt { | ||||
|   PRE_NOTIFY, | ||||
| }; | ||||
|  | ||||
| // Class to make sure only one BLECharacteristicSetValueAction is active at a time for each characteristic | ||||
| class BLECharacteristicSetValueActionManager | ||||
|     : public EventEmitter<BLECharacteristicSetValueActionEvt, BLECharacteristic *> { | ||||
| class BLECharacteristicSetValueActionManager { | ||||
|  public: | ||||
|   // Singleton pattern | ||||
|   static BLECharacteristicSetValueActionManager *get_instance() { | ||||
|     static BLECharacteristicSetValueActionManager instance; | ||||
|     return &instance; | ||||
|   } | ||||
|   void set_listener(BLECharacteristic *characteristic, EventEmitterListenerID listener_id, | ||||
|                     const std::function<void()> &pre_notify_listener); | ||||
|   EventEmitterListenerID get_listener(BLECharacteristic *characteristic) { | ||||
|   void set_listener(BLECharacteristic *characteristic, const std::function<void()> &pre_notify_listener); | ||||
|   bool has_listener(BLECharacteristic *characteristic) { return this->find_listener_(characteristic) != nullptr; } | ||||
|   void emit_pre_notify(BLECharacteristic *characteristic) { | ||||
|     for (const auto &entry : this->listeners_) { | ||||
|       if (entry.characteristic == characteristic) { | ||||
|         return entry.listener_id; | ||||
|         entry.pre_notify_listener(); | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|     return INVALID_LISTENER_ID; | ||||
|   } | ||||
|   void emit_pre_notify(BLECharacteristic *characteristic) { | ||||
|     this->emit_(BLECharacteristicSetValueActionEvt::PRE_NOTIFY, characteristic); | ||||
|   } | ||||
|  | ||||
|  private: | ||||
|   struct ListenerEntry { | ||||
|     BLECharacteristic *characteristic; | ||||
|     EventEmitterListenerID listener_id; | ||||
|     EventEmitterListenerID pre_notify_listener_id; | ||||
|     std::function<void()> pre_notify_listener; | ||||
|   }; | ||||
|   std::vector<ListenerEntry> listeners_; | ||||
|  | ||||
| @@ -87,24 +73,22 @@ template<typename... Ts> class BLECharacteristicSetValueAction : public Action<T | ||||
|   void set_buffer(ByteBuffer buffer) { this->set_buffer(buffer.get_data()); } | ||||
|   void play(Ts... x) override { | ||||
|     // If the listener is already set, do nothing | ||||
|     if (BLECharacteristicSetValueActionManager::get_instance()->get_listener(this->parent_) == this->listener_id_) | ||||
|     if (BLECharacteristicSetValueActionManager::get_instance()->has_listener(this->parent_)) | ||||
|       return; | ||||
|     // Set initial value | ||||
|     this->parent_->set_value(this->buffer_.value(x...)); | ||||
|     // Set the listener for read events | ||||
|     this->listener_id_ = this->parent_->EventEmitter<BLECharacteristicEvt::EmptyEvt, uint16_t>::on( | ||||
|         BLECharacteristicEvt::EmptyEvt::ON_READ, [this, x...](uint16_t id) { | ||||
|     this->parent_->on_read([this, x...](uint16_t id) { | ||||
|       // Set the value of the characteristic every time it is read | ||||
|       this->parent_->set_value(this->buffer_.value(x...)); | ||||
|     }); | ||||
|     // Set the listener in the global manager so only one BLECharacteristicSetValueAction is set for each characteristic | ||||
|     BLECharacteristicSetValueActionManager::get_instance()->set_listener( | ||||
|         this->parent_, this->listener_id_, [this, x...]() { this->parent_->set_value(this->buffer_.value(x...)); }); | ||||
|         this->parent_, [this, x...]() { this->parent_->set_value(this->buffer_.value(x...)); }); | ||||
|   } | ||||
|  | ||||
|  protected: | ||||
|   BLECharacteristic *parent_; | ||||
|   EventEmitterListenerID listener_id_; | ||||
| }; | ||||
| #endif  // USE_ESP32_BLE_SERVER_SET_VALUE_ACTION | ||||
|  | ||||
|   | ||||
| @@ -1,14 +1,13 @@ | ||||
| from __future__ import annotations | ||||
|  | ||||
| from collections.abc import Callable, MutableMapping | ||||
| import logging | ||||
| from typing import Any | ||||
|  | ||||
| from esphome import automation | ||||
| import esphome.codegen as cg | ||||
| from esphome.components import esp32_ble | ||||
| from esphome.components.esp32 import add_idf_sdkconfig_option | ||||
| from esphome.components.esp32_ble import ( | ||||
|     IDF_MAX_CONNECTIONS, | ||||
|     BTLoggers, | ||||
|     bt_uuid, | ||||
|     bt_uuid16_format, | ||||
| @@ -24,6 +23,7 @@ from esphome.const import ( | ||||
|     CONF_INTERVAL, | ||||
|     CONF_MAC_ADDRESS, | ||||
|     CONF_MANUFACTURER_ID, | ||||
|     CONF_MAX_CONNECTIONS, | ||||
|     CONF_ON_BLE_ADVERTISE, | ||||
|     CONF_ON_BLE_MANUFACTURER_DATA_ADVERTISE, | ||||
|     CONF_ON_BLE_SERVICE_DATA_ADVERTISE, | ||||
| @@ -38,19 +38,12 @@ AUTO_LOAD = ["esp32_ble"] | ||||
| DEPENDENCIES = ["esp32"] | ||||
| CODEOWNERS = ["@bdraco"] | ||||
|  | ||||
| KEY_ESP32_BLE_TRACKER = "esp32_ble_tracker" | ||||
| KEY_USED_CONNECTION_SLOTS = "used_connection_slots" | ||||
|  | ||||
| CONF_MAX_CONNECTIONS = "max_connections" | ||||
| CONF_ESP32_BLE_ID = "esp32_ble_id" | ||||
| CONF_SCAN_PARAMETERS = "scan_parameters" | ||||
| CONF_WINDOW = "window" | ||||
| CONF_ON_SCAN_END = "on_scan_end" | ||||
| CONF_SOFTWARE_COEXISTENCE = "software_coexistence" | ||||
|  | ||||
| DEFAULT_MAX_CONNECTIONS = 3 | ||||
| IDF_MAX_CONNECTIONS = 9 | ||||
|  | ||||
| _LOGGER = logging.getLogger(__name__) | ||||
|  | ||||
|  | ||||
| @@ -128,6 +121,15 @@ def validate_scan_parameters(config): | ||||
|     return config | ||||
|  | ||||
|  | ||||
| def validate_max_connections_deprecated(config: ConfigType) -> ConfigType: | ||||
|     if CONF_MAX_CONNECTIONS in config: | ||||
|         _LOGGER.warning( | ||||
|             "The 'max_connections' option in 'esp32_ble_tracker' is deprecated. " | ||||
|             "Please move it to the 'esp32_ble' component instead." | ||||
|         ) | ||||
|     return config | ||||
|  | ||||
|  | ||||
| def as_hex(value): | ||||
|     return cg.RawExpression(f"0x{value}ULL") | ||||
|  | ||||
| @@ -150,24 +152,12 @@ def as_reversed_hex_array(value): | ||||
|     ) | ||||
|  | ||||
|  | ||||
| def consume_connection_slots( | ||||
|     value: int, consumer: str | ||||
| ) -> Callable[[MutableMapping], MutableMapping]: | ||||
|     def _consume_connection_slots(config: MutableMapping) -> MutableMapping: | ||||
|         data: dict[str, Any] = CORE.data.setdefault(KEY_ESP32_BLE_TRACKER, {}) | ||||
|         slots: list[str] = data.setdefault(KEY_USED_CONNECTION_SLOTS, []) | ||||
|         slots.extend([consumer] * value) | ||||
|         return config | ||||
|  | ||||
|     return _consume_connection_slots | ||||
|  | ||||
|  | ||||
| CONFIG_SCHEMA = cv.All( | ||||
|     cv.Schema( | ||||
|         { | ||||
|             cv.GenerateID(): cv.declare_id(ESP32BLETracker), | ||||
|             cv.GenerateID(esp32_ble.CONF_BLE_ID): cv.use_id(esp32_ble.ESP32BLE), | ||||
|             cv.Optional(CONF_MAX_CONNECTIONS, default=DEFAULT_MAX_CONNECTIONS): cv.All( | ||||
|             cv.Optional(CONF_MAX_CONNECTIONS): cv.All( | ||||
|                 cv.positive_int, cv.Range(min=0, max=IDF_MAX_CONNECTIONS) | ||||
|             ), | ||||
|             cv.Optional(CONF_SCAN_PARAMETERS, default={}): cv.All( | ||||
| @@ -224,48 +214,11 @@ CONFIG_SCHEMA = cv.All( | ||||
|             cv.OnlyWith(CONF_SOFTWARE_COEXISTENCE, "wifi", default=True): bool, | ||||
|         } | ||||
|     ).extend(cv.COMPONENT_SCHEMA), | ||||
|     validate_max_connections_deprecated, | ||||
| ) | ||||
|  | ||||
|  | ||||
| def validate_remaining_connections(config): | ||||
|     data: dict[str, Any] = CORE.data.get(KEY_ESP32_BLE_TRACKER, {}) | ||||
|     slots: list[str] = data.get(KEY_USED_CONNECTION_SLOTS, []) | ||||
|     used_slots = len(slots) | ||||
|     if used_slots <= config[CONF_MAX_CONNECTIONS]: | ||||
|         return config | ||||
|     slot_users = ", ".join(slots) | ||||
|  | ||||
|     if used_slots < IDF_MAX_CONNECTIONS: | ||||
|         _LOGGER.warning( | ||||
|             "esp32_ble_tracker exceeded `%s`: components attempted to consume %d " | ||||
|             "connection slot(s) out of available configured maximum %d connection " | ||||
|             "slot(s); The system automatically increased `%s` to %d to match the " | ||||
|             "number of used connection slot(s) by components: %s.", | ||||
|             CONF_MAX_CONNECTIONS, | ||||
|             used_slots, | ||||
|             config[CONF_MAX_CONNECTIONS], | ||||
|             CONF_MAX_CONNECTIONS, | ||||
|             used_slots, | ||||
|             slot_users, | ||||
|         ) | ||||
|         config[CONF_MAX_CONNECTIONS] = used_slots | ||||
|         return config | ||||
|  | ||||
|     msg = ( | ||||
|         f"esp32_ble_tracker exceeded `{CONF_MAX_CONNECTIONS}`: " | ||||
|         f"components attempted to consume {used_slots} connection slot(s) " | ||||
|         f"out of available configured maximum {config[CONF_MAX_CONNECTIONS]} " | ||||
|         f"connection slot(s); Decrease the number of BLE clients ({slot_users})" | ||||
|     ) | ||||
|     if config[CONF_MAX_CONNECTIONS] < IDF_MAX_CONNECTIONS: | ||||
|         msg += f" or increase {CONF_MAX_CONNECTIONS}` to {used_slots}" | ||||
|     msg += f" to stay under the {IDF_MAX_CONNECTIONS} connection slot(s) limit." | ||||
|     raise cv.Invalid(msg) | ||||
|  | ||||
|  | ||||
| FINAL_VALIDATE_SCHEMA = cv.All( | ||||
|     validate_remaining_connections, esp32_ble.validate_variant | ||||
| ) | ||||
| FINAL_VALIDATE_SCHEMA = esp32_ble.validate_variant | ||||
|  | ||||
| ESP_BLE_DEVICE_SCHEMA = cv.Schema( | ||||
|     { | ||||
| @@ -345,10 +298,8 @@ async def to_code(config): | ||||
|     # Match arduino CONFIG_BTU_TASK_STACK_SIZE | ||||
|     # https://github.com/espressif/arduino-esp32/blob/fd72cf46ad6fc1a6de99c1d83ba8eba17d80a4ee/tools/sdk/esp32/sdkconfig#L1866 | ||||
|     add_idf_sdkconfig_option("CONFIG_BT_BTU_TASK_STACK_SIZE", 8192) | ||||
|     add_idf_sdkconfig_option("CONFIG_BT_ACL_CONNECTIONS", 9) | ||||
|     add_idf_sdkconfig_option( | ||||
|         "CONFIG_BTDM_CTRL_BLE_MAX_CONN", config[CONF_MAX_CONNECTIONS] | ||||
|     ) | ||||
|     # Note: CONFIG_BT_ACL_CONNECTIONS and CONFIG_BTDM_CTRL_BLE_MAX_CONN are now | ||||
|     # configured in esp32_ble component based on max_connections setting | ||||
|  | ||||
|     cg.add_define("USE_OTA_STATE_CALLBACK")  # To be notified when an OTA update starts | ||||
|     cg.add_define("USE_ESP32_BLE_CLIENT") | ||||
|   | ||||
| @@ -67,8 +67,16 @@ static bool get_bitrate(canbus::CanSpeed bitrate, twai_timing_config_t *t_config | ||||
| } | ||||
|  | ||||
| bool ESP32Can::setup_internal() { | ||||
|   static int next_twai_ctrl_num = 0; | ||||
|   if (static_cast<unsigned>(next_twai_ctrl_num) >= SOC_TWAI_CONTROLLER_NUM) { | ||||
|     ESP_LOGW(TAG, "Maximum number of esp32_can components created already"); | ||||
|     this->mark_failed(); | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   twai_general_config_t g_config = | ||||
|       TWAI_GENERAL_CONFIG_DEFAULT((gpio_num_t) this->tx_, (gpio_num_t) this->rx_, TWAI_MODE_NORMAL); | ||||
|   g_config.controller_id = next_twai_ctrl_num++; | ||||
|   if (this->tx_queue_len_.has_value()) { | ||||
|     g_config.tx_queue_len = this->tx_queue_len_.value(); | ||||
|   } | ||||
| @@ -86,14 +94,14 @@ bool ESP32Can::setup_internal() { | ||||
|   } | ||||
|  | ||||
|   // Install TWAI driver | ||||
|   if (twai_driver_install(&g_config, &t_config, &f_config) != ESP_OK) { | ||||
|   if (twai_driver_install_v2(&g_config, &t_config, &f_config, &(this->twai_handle_)) != ESP_OK) { | ||||
|     // Failed to install driver | ||||
|     this->mark_failed(); | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   // Start TWAI driver | ||||
|   if (twai_start() != ESP_OK) { | ||||
|   if (twai_start_v2(this->twai_handle_) != ESP_OK) { | ||||
|     // Failed to start driver | ||||
|     this->mark_failed(); | ||||
|     return false; | ||||
| @@ -102,6 +110,11 @@ bool ESP32Can::setup_internal() { | ||||
| } | ||||
|  | ||||
| canbus::Error ESP32Can::send_message(struct canbus::CanFrame *frame) { | ||||
|   if (this->twai_handle_ == nullptr) { | ||||
|     // not setup yet or setup failed | ||||
|     return canbus::ERROR_FAIL; | ||||
|   } | ||||
|  | ||||
|   if (frame->can_data_length_code > canbus::CAN_MAX_DATA_LENGTH) { | ||||
|     return canbus::ERROR_FAILTX; | ||||
|   } | ||||
| @@ -124,7 +137,7 @@ canbus::Error ESP32Can::send_message(struct canbus::CanFrame *frame) { | ||||
|     memcpy(message.data, frame->data, frame->can_data_length_code); | ||||
|   } | ||||
|  | ||||
|   if (twai_transmit(&message, this->tx_enqueue_timeout_ticks_) == ESP_OK) { | ||||
|   if (twai_transmit_v2(this->twai_handle_, &message, this->tx_enqueue_timeout_ticks_) == ESP_OK) { | ||||
|     return canbus::ERROR_OK; | ||||
|   } else { | ||||
|     return canbus::ERROR_ALLTXBUSY; | ||||
| @@ -132,9 +145,14 @@ canbus::Error ESP32Can::send_message(struct canbus::CanFrame *frame) { | ||||
| } | ||||
|  | ||||
| canbus::Error ESP32Can::read_message(struct canbus::CanFrame *frame) { | ||||
|   if (this->twai_handle_ == nullptr) { | ||||
|     // not setup yet or setup failed | ||||
|     return canbus::ERROR_FAIL; | ||||
|   } | ||||
|  | ||||
|   twai_message_t message; | ||||
|  | ||||
|   if (twai_receive(&message, 0) != ESP_OK) { | ||||
|   if (twai_receive_v2(this->twai_handle_, &message, 0) != ESP_OK) { | ||||
|     return canbus::ERROR_NOMSG; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -5,6 +5,8 @@ | ||||
| #include "esphome/components/canbus/canbus.h" | ||||
| #include "esphome/core/component.h" | ||||
|  | ||||
| #include <driver/twai.h> | ||||
|  | ||||
| namespace esphome { | ||||
| namespace esp32_can { | ||||
|  | ||||
| @@ -29,6 +31,7 @@ class ESP32Can : public canbus::Canbus { | ||||
|   TickType_t tx_enqueue_timeout_ticks_{}; | ||||
|   optional<uint32_t> tx_queue_len_{}; | ||||
|   optional<uint32_t> rx_queue_len_{}; | ||||
|   twai_handle_t twai_handle_{nullptr}; | ||||
| }; | ||||
|  | ||||
| }  // namespace esp32_can | ||||
|   | ||||
| @@ -38,8 +38,7 @@ void ESP32ImprovComponent::setup() { | ||||
|     }); | ||||
|   } | ||||
| #endif | ||||
|   global_ble_server->on(BLEServerEvt::EmptyEvt::ON_DISCONNECT, | ||||
|                         [this](uint16_t conn_id) { this->set_error_(improv::ERROR_NONE); }); | ||||
|   global_ble_server->on_disconnect([this](uint16_t conn_id) { this->set_error_(improv::ERROR_NONE); }); | ||||
|  | ||||
|   // Start with loop disabled - will be enabled by start() when needed | ||||
|   this->disable_loop(); | ||||
| @@ -57,8 +56,7 @@ void ESP32ImprovComponent::setup_characteristics() { | ||||
|   this->error_->add_descriptor(error_descriptor); | ||||
|  | ||||
|   this->rpc_ = this->service_->create_characteristic(improv::RPC_COMMAND_UUID, BLECharacteristic::PROPERTY_WRITE); | ||||
|   this->rpc_->EventEmitter<BLECharacteristicEvt::VectorEvt, std::vector<uint8_t>, uint16_t>::on( | ||||
|       BLECharacteristicEvt::VectorEvt::ON_WRITE, [this](const std::vector<uint8_t> &data, uint16_t id) { | ||||
|   this->rpc_->on_write([this](std::span<const uint8_t> data, uint16_t id) { | ||||
|     if (!data.empty()) { | ||||
|       this->incoming_data_.insert(this->incoming_data_.end(), data.begin(), data.end()); | ||||
|     } | ||||
|   | ||||
| @@ -35,7 +35,7 @@ static size_t IRAM_ATTR HOT encoder_callback(const void *data, size_t size, size | ||||
|     if (symbols_free < RMT_SYMBOLS_PER_BYTE) { | ||||
|       return 0; | ||||
|     } | ||||
|     for (int32_t i = 0; i < RMT_SYMBOLS_PER_BYTE; i++) { | ||||
|     for (size_t i = 0; i < RMT_SYMBOLS_PER_BYTE; i++) { | ||||
|       if (bytes[index] & (1 << (7 - i))) { | ||||
|         symbols[i] = params->bit1; | ||||
|       } else { | ||||
|   | ||||
| @@ -614,24 +614,67 @@ bool ESPHomeOTAComponent::handle_auth_send_() { | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
|     // Generate nonce with appropriate hasher | ||||
|     bool success = false; | ||||
|     // Generate nonce - hasher must be created and used in same stack frame | ||||
|     // CRITICAL ESP32-S3 HARDWARE SHA ACCELERATION REQUIREMENTS: | ||||
|     // 1. Hash objects must NEVER be passed to another function (different stack frame) | ||||
|     // 2. NO Variable Length Arrays (VLAs) - they corrupt the stack with hardware DMA | ||||
|     // 3. All hash operations (init/add/calculate) must happen in the SAME function where object is created | ||||
|     // Violating these causes truncated hash output (20 bytes instead of 32) or memory corruption. | ||||
|     // | ||||
|     // Buffer layout after AUTH_READ completes: | ||||
|     //   [0]: auth_type (1 byte) | ||||
|     //   [1...hex_size]: nonce (hex_size bytes) - our random nonce sent in AUTH_SEND | ||||
|     //   [1+hex_size...1+2*hex_size-1]: cnonce (hex_size bytes) - client's nonce | ||||
|     //   [1+2*hex_size...1+3*hex_size-1]: response (hex_size bytes) - client's hash | ||||
|  | ||||
|     // Declare both hash objects in same stack frame, use pointer to select. | ||||
|     // NOTE: Both objects are declared here even though only one is used. This is REQUIRED for ESP32-S3 | ||||
|     // hardware SHA acceleration - the object must exist in this stack frame for all operations. | ||||
|     // Do NOT try to "optimize" by creating the object inside the if block, as it would go out of scope. | ||||
| #ifdef USE_OTA_SHA256 | ||||
|     sha256::SHA256 sha_hasher; | ||||
| #endif | ||||
| #ifdef USE_OTA_MD5 | ||||
|     md5::MD5Digest md5_hasher; | ||||
| #endif | ||||
|     HashBase *hasher = nullptr; | ||||
|  | ||||
| #ifdef USE_OTA_SHA256 | ||||
|     if (this->auth_type_ == ota::OTA_RESPONSE_REQUEST_SHA256_AUTH) { | ||||
|       sha256::SHA256 sha_hasher; | ||||
|       success = this->prepare_auth_nonce_(&sha_hasher); | ||||
|       hasher = &sha_hasher; | ||||
|     } | ||||
| #endif | ||||
| #ifdef USE_OTA_MD5 | ||||
|     if (this->auth_type_ == ota::OTA_RESPONSE_REQUEST_AUTH) { | ||||
|       md5::MD5Digest md5_hasher; | ||||
|       success = this->prepare_auth_nonce_(&md5_hasher); | ||||
|       hasher = &md5_hasher; | ||||
|     } | ||||
| #endif | ||||
|  | ||||
|     if (!success) { | ||||
|     const size_t hex_size = hasher->get_size() * 2; | ||||
|     const size_t nonce_len = hasher->get_size() / 4; | ||||
|     const size_t auth_buf_size = 1 + 3 * hex_size; | ||||
|     this->auth_buf_ = std::make_unique<uint8_t[]>(auth_buf_size); | ||||
|     this->auth_buf_pos_ = 0; | ||||
|  | ||||
|     char *buf = reinterpret_cast<char *>(this->auth_buf_.get() + 1); | ||||
|     if (!random_bytes(reinterpret_cast<uint8_t *>(buf), nonce_len)) { | ||||
|       this->log_auth_warning_(LOG_STR("Random failed")); | ||||
|       this->send_error_and_cleanup_(ota::OTA_RESPONSE_ERROR_UNKNOWN); | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
|     hasher->init(); | ||||
|     hasher->add(buf, nonce_len); | ||||
|     hasher->calculate(); | ||||
|     this->auth_buf_[0] = this->auth_type_; | ||||
|     hasher->get_hex(buf); | ||||
|  | ||||
| #if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE | ||||
|     char log_buf[65];  // Fixed size for SHA256 hex (64) + null, works for MD5 (32) too | ||||
|     memcpy(log_buf, buf, hex_size); | ||||
|     log_buf[hex_size] = '\0'; | ||||
|     ESP_LOGV(TAG, "Auth: Nonce is %s", log_buf); | ||||
| #endif | ||||
|   } | ||||
|  | ||||
|   // Try to write auth_type + nonce | ||||
| @@ -678,89 +721,41 @@ bool ESPHomeOTAComponent::handle_auth_read_() { | ||||
|   } | ||||
|  | ||||
|   // We have all the data, verify it | ||||
|   bool matches = false; | ||||
|   const char *nonce = reinterpret_cast<char *>(this->auth_buf_.get() + 1); | ||||
|   const char *cnonce = nonce + hex_size; | ||||
|   const char *response = cnonce + hex_size; | ||||
|  | ||||
|   // CRITICAL ESP32-S3: Hash objects must stay in same stack frame (no passing to other functions). | ||||
|   // Declare both hash objects in same stack frame, use pointer to select. | ||||
|   // NOTE: Both objects are declared here even though only one is used. This is REQUIRED for ESP32-S3 | ||||
|   // hardware SHA acceleration - the object must exist in this stack frame for all operations. | ||||
|   // Do NOT try to "optimize" by creating the object inside the if block, as it would go out of scope. | ||||
| #ifdef USE_OTA_SHA256 | ||||
|   sha256::SHA256 sha_hasher; | ||||
| #endif | ||||
| #ifdef USE_OTA_MD5 | ||||
|   md5::MD5Digest md5_hasher; | ||||
| #endif | ||||
|   HashBase *hasher = nullptr; | ||||
|  | ||||
| #ifdef USE_OTA_SHA256 | ||||
|   if (this->auth_type_ == ota::OTA_RESPONSE_REQUEST_SHA256_AUTH) { | ||||
|     sha256::SHA256 sha_hasher; | ||||
|     matches = this->verify_hash_auth_(&sha_hasher, hex_size); | ||||
|     hasher = &sha_hasher; | ||||
|   } | ||||
| #endif | ||||
| #ifdef USE_OTA_MD5 | ||||
|   if (this->auth_type_ == ota::OTA_RESPONSE_REQUEST_AUTH) { | ||||
|     md5::MD5Digest md5_hasher; | ||||
|     matches = this->verify_hash_auth_(&md5_hasher, hex_size); | ||||
|     hasher = &md5_hasher; | ||||
|   } | ||||
| #endif | ||||
|  | ||||
|   if (!matches) { | ||||
|     this->log_auth_warning_(LOG_STR("Password mismatch")); | ||||
|     this->send_error_and_cleanup_(ota::OTA_RESPONSE_ERROR_AUTH_INVALID); | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   // Authentication successful - clean up auth state | ||||
|   this->cleanup_auth_(); | ||||
|  | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| bool ESPHomeOTAComponent::prepare_auth_nonce_(HashBase *hasher) { | ||||
|   // Calculate required buffer size using the hasher | ||||
|   const size_t hex_size = hasher->get_size() * 2; | ||||
|   const size_t nonce_len = hasher->get_size() / 4; | ||||
|  | ||||
|   // Buffer layout after AUTH_READ completes: | ||||
|   //   [0]: auth_type (1 byte) | ||||
|   //   [1...hex_size]: nonce (hex_size bytes) - our random nonce sent in AUTH_SEND | ||||
|   //   [1+hex_size...1+2*hex_size-1]: cnonce (hex_size bytes) - client's nonce | ||||
|   //   [1+2*hex_size...1+3*hex_size-1]: response (hex_size bytes) - client's hash | ||||
|   // Total: 1 + 3*hex_size | ||||
|   const size_t auth_buf_size = 1 + 3 * hex_size; | ||||
|   this->auth_buf_ = std::make_unique<uint8_t[]>(auth_buf_size); | ||||
|   this->auth_buf_pos_ = 0; | ||||
|  | ||||
|   // Generate nonce | ||||
|   char *buf = reinterpret_cast<char *>(this->auth_buf_.get() + 1); | ||||
|   if (!random_bytes(reinterpret_cast<uint8_t *>(buf), nonce_len)) { | ||||
|     this->log_auth_warning_(LOG_STR("Random failed")); | ||||
|     this->send_error_and_cleanup_(ota::OTA_RESPONSE_ERROR_UNKNOWN); | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   hasher->init(); | ||||
|   hasher->add(buf, nonce_len); | ||||
|   hasher->calculate(); | ||||
|  | ||||
|   // Prepare buffer: auth_type (1 byte) + nonce (hex_size bytes) | ||||
|   this->auth_buf_[0] = this->auth_type_; | ||||
|   hasher->get_hex(buf); | ||||
|  | ||||
| #if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE | ||||
|   char log_buf[hex_size + 1]; | ||||
|   // Log nonce for debugging | ||||
|   memcpy(log_buf, buf, hex_size); | ||||
|   log_buf[hex_size] = '\0'; | ||||
|   ESP_LOGV(TAG, "Auth: Nonce is %s", log_buf); | ||||
| #endif | ||||
|  | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| bool ESPHomeOTAComponent::verify_hash_auth_(HashBase *hasher, size_t hex_size) { | ||||
|   // Get pointers to the data in the buffer (see prepare_auth_nonce_ for buffer layout) | ||||
|   const char *nonce = reinterpret_cast<char *>(this->auth_buf_.get() + 1);  // Skip auth_type byte | ||||
|   const char *cnonce = nonce + hex_size;                                    // CNonce immediately follows nonce | ||||
|   const char *response = cnonce + hex_size;                                 // Response immediately follows cnonce | ||||
|  | ||||
|   // Calculate expected hash: password + nonce + cnonce | ||||
|   hasher->init(); | ||||
|   hasher->add(this->password_.c_str(), this->password_.length()); | ||||
|   hasher->add(nonce, hex_size * 2);  // Add both nonce and cnonce (contiguous in buffer) | ||||
|   hasher->calculate(); | ||||
|  | ||||
| #if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE | ||||
|   char log_buf[hex_size + 1]; | ||||
|   char log_buf[65];  // Fixed size for SHA256 hex (64) + null, works for MD5 (32) too | ||||
|   // Log CNonce | ||||
|   memcpy(log_buf, cnonce, hex_size); | ||||
|   log_buf[hex_size] = '\0'; | ||||
| @@ -778,7 +773,18 @@ bool ESPHomeOTAComponent::verify_hash_auth_(HashBase *hasher, size_t hex_size) { | ||||
| #endif | ||||
|  | ||||
|   // Compare response | ||||
|   return hasher->equals_hex(response); | ||||
|   bool matches = hasher->equals_hex(response); | ||||
|  | ||||
|   if (!matches) { | ||||
|     this->log_auth_warning_(LOG_STR("Password mismatch")); | ||||
|     this->send_error_and_cleanup_(ota::OTA_RESPONSE_ERROR_AUTH_INVALID); | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   // Authentication successful - clean up auth state | ||||
|   this->cleanup_auth_(); | ||||
|  | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| size_t ESPHomeOTAComponent::get_auth_hex_size_() const { | ||||
|   | ||||
| @@ -47,8 +47,6 @@ class ESPHomeOTAComponent : public ota::OTAComponent { | ||||
|   bool handle_auth_send_(); | ||||
|   bool handle_auth_read_(); | ||||
|   bool select_auth_type_(); | ||||
|   bool prepare_auth_nonce_(HashBase *hasher); | ||||
|   bool verify_hash_auth_(HashBase *hasher, size_t hex_size); | ||||
|   size_t get_auth_hex_size_() const; | ||||
|   void cleanup_auth_(); | ||||
|   void log_auth_warning_(const LogString *msg); | ||||
|   | ||||
| @@ -41,17 +41,20 @@ static const char *const TAG = "ethernet"; | ||||
|  | ||||
| EthernetComponent *global_eth_component;  // NOLINT(cppcoreguidelines-avoid-non-const-global-variables) | ||||
|  | ||||
| void EthernetComponent::log_error_and_mark_failed_(esp_err_t err, const char *message) { | ||||
|   ESP_LOGE(TAG, "%s: (%d) %s", message, err, esp_err_to_name(err)); | ||||
|   this->mark_failed(); | ||||
| } | ||||
|  | ||||
| #define ESPHL_ERROR_CHECK(err, message) \ | ||||
|   if ((err) != ESP_OK) { \ | ||||
|     ESP_LOGE(TAG, message ": (%d) %s", err, esp_err_to_name(err)); \ | ||||
|     this->mark_failed(); \ | ||||
|     this->log_error_and_mark_failed_(err, message); \ | ||||
|     return; \ | ||||
|   } | ||||
|  | ||||
| #define ESPHL_ERROR_CHECK_RET(err, message, ret) \ | ||||
|   if ((err) != ESP_OK) { \ | ||||
|     ESP_LOGE(TAG, message ": (%d) %s", err, esp_err_to_name(err)); \ | ||||
|     this->mark_failed(); \ | ||||
|     this->log_error_and_mark_failed_(err, message); \ | ||||
|     return ret; \ | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -106,6 +106,7 @@ class EthernetComponent : public Component { | ||||
|   void start_connect_(); | ||||
|   void finish_connect_(); | ||||
|   void dump_connect_params_(); | ||||
|   void log_error_and_mark_failed_(esp_err_t err, const char *message); | ||||
| #ifdef USE_ETHERNET_KSZ8081 | ||||
|   /// @brief Set `RMII Reference Clock Select` bit for KSZ8081. | ||||
|   void ksz8081_set_clock_reference_(esp_eth_mac_t *mac); | ||||
| @@ -162,7 +163,7 @@ class EthernetComponent : public Component { | ||||
| // NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables) | ||||
| extern EthernetComponent *global_eth_component; | ||||
|  | ||||
| #if defined(USE_ARDUINO) || ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 4, 2) | ||||
| #if ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 4, 2) | ||||
| extern "C" esp_eth_phy_t *esp_eth_phy_new_jl1101(const eth_phy_config_t *config); | ||||
| #endif | ||||
|  | ||||
|   | ||||
| @@ -1,5 +0,0 @@ | ||||
| CODEOWNERS = ["@Rapsssito"] | ||||
|  | ||||
| # Allows event_emitter to be configured in yaml, to allow use of the C++ api. | ||||
|  | ||||
| CONFIG_SCHEMA = {} | ||||
| @@ -1,117 +0,0 @@ | ||||
| #pragma once | ||||
| #include <vector> | ||||
| #include <functional> | ||||
| #include <limits> | ||||
|  | ||||
| #include "esphome/core/log.h" | ||||
|  | ||||
| namespace esphome { | ||||
| namespace event_emitter { | ||||
|  | ||||
| using EventEmitterListenerID = uint32_t; | ||||
| static constexpr EventEmitterListenerID INVALID_LISTENER_ID = 0; | ||||
|  | ||||
| // EventEmitter class that can emit events with a specific name (it is highly recommended to use an enum class for this) | ||||
| // and a list of arguments. Supports multiple listeners for each event. | ||||
| template<typename EvtType, typename... Args> class EventEmitter { | ||||
|  public: | ||||
|   EventEmitterListenerID on(EvtType event, std::function<void(Args...)> listener) { | ||||
|     EventEmitterListenerID listener_id = this->get_next_id_(); | ||||
|  | ||||
|     // Find or create event entry | ||||
|     EventEntry *entry = this->find_or_create_event_(event); | ||||
|     entry->listeners.push_back({listener_id, listener}); | ||||
|  | ||||
|     return listener_id; | ||||
|   } | ||||
|  | ||||
|   void off(EvtType event, EventEmitterListenerID id) { | ||||
|     EventEntry *entry = this->find_event_(event); | ||||
|     if (entry == nullptr) | ||||
|       return; | ||||
|  | ||||
|     // Remove listener with given id | ||||
|     for (auto it = entry->listeners.begin(); it != entry->listeners.end(); ++it) { | ||||
|       if (it->id == id) { | ||||
|         // Swap with last and pop for efficient removal | ||||
|         *it = entry->listeners.back(); | ||||
|         entry->listeners.pop_back(); | ||||
|  | ||||
|         // Remove event entry if no more listeners | ||||
|         if (entry->listeners.empty()) { | ||||
|           this->remove_event_(event); | ||||
|         } | ||||
|         return; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|  protected: | ||||
|   void emit_(EvtType event, Args... args) { | ||||
|     EventEntry *entry = this->find_event_(event); | ||||
|     if (entry == nullptr) | ||||
|       return; | ||||
|  | ||||
|     // Call all listeners for this event | ||||
|     for (const auto &listener : entry->listeners) { | ||||
|       listener.callback(args...); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|  private: | ||||
|   struct Listener { | ||||
|     EventEmitterListenerID id; | ||||
|     std::function<void(Args...)> callback; | ||||
|   }; | ||||
|  | ||||
|   struct EventEntry { | ||||
|     EvtType event; | ||||
|     std::vector<Listener> listeners; | ||||
|   }; | ||||
|  | ||||
|   EventEmitterListenerID get_next_id_() { | ||||
|     // Simple incrementing ID, wrapping around at max | ||||
|     EventEmitterListenerID next_id = (this->current_id_ + 1); | ||||
|     if (next_id == INVALID_LISTENER_ID) { | ||||
|       next_id = 1; | ||||
|     } | ||||
|     this->current_id_ = next_id; | ||||
|     return this->current_id_; | ||||
|   } | ||||
|  | ||||
|   EventEntry *find_event_(EvtType event) { | ||||
|     for (auto &entry : this->events_) { | ||||
|       if (entry.event == event) { | ||||
|         return &entry; | ||||
|       } | ||||
|     } | ||||
|     return nullptr; | ||||
|   } | ||||
|  | ||||
|   EventEntry *find_or_create_event_(EvtType event) { | ||||
|     EventEntry *entry = this->find_event_(event); | ||||
|     if (entry != nullptr) | ||||
|       return entry; | ||||
|  | ||||
|     // Create new event entry | ||||
|     this->events_.push_back({event, {}}); | ||||
|     return &this->events_.back(); | ||||
|   } | ||||
|  | ||||
|   void remove_event_(EvtType event) { | ||||
|     for (auto it = this->events_.begin(); it != this->events_.end(); ++it) { | ||||
|       if (it->event == event) { | ||||
|         // Swap with last and pop | ||||
|         *it = this->events_.back(); | ||||
|         this->events_.pop_back(); | ||||
|         return; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   std::vector<EventEntry> events_; | ||||
|   EventEmitterListenerID current_id_ = 0; | ||||
| }; | ||||
|  | ||||
| }  // namespace event_emitter | ||||
| }  // namespace esphome | ||||
| @@ -80,7 +80,7 @@ void FingerprintGrowComponent::setup() { | ||||
|   delay(20);  // This delay guarantees the sensor will in fact be powered power. | ||||
|  | ||||
|   if (this->check_password_()) { | ||||
|     if (this->new_password_ != -1) { | ||||
|     if (this->new_password_ != std::numeric_limits<uint32_t>::max()) { | ||||
|       if (this->set_password_()) | ||||
|         return; | ||||
|     } else { | ||||
|   | ||||
| @@ -6,6 +6,7 @@ | ||||
| #include "esphome/components/binary_sensor/binary_sensor.h" | ||||
| #include "esphome/components/uart/uart.h" | ||||
|  | ||||
| #include <limits> | ||||
| #include <vector> | ||||
|  | ||||
| namespace esphome { | ||||
| @@ -177,7 +178,7 @@ class FingerprintGrowComponent : public PollingComponent, public uart::UARTDevic | ||||
|   uint8_t address_[4] = {0xFF, 0xFF, 0xFF, 0xFF}; | ||||
|   uint16_t capacity_ = 64; | ||||
|   uint32_t password_ = 0x0; | ||||
|   uint32_t new_password_ = -1; | ||||
|   uint32_t new_password_ = std::numeric_limits<uint32_t>::max(); | ||||
|   GPIOPin *sensing_pin_{nullptr}; | ||||
|   GPIOPin *sensor_power_pin_{nullptr}; | ||||
|   uint8_t enrollment_image_ = 0; | ||||
|   | ||||
| @@ -179,7 +179,7 @@ void Graph::draw(Display *buff, uint16_t x_offset, uint16_t y_offset, Color colo | ||||
|         if (b) { | ||||
|           int16_t y = (int16_t) roundf((this->height_ - 1) * (1.0 - v)) - thick / 2 + y_offset; | ||||
|           auto draw_pixel_at = [&buff, c, y_offset, this](int16_t x, int16_t y) { | ||||
|             if (y >= y_offset && y < y_offset + this->height_) | ||||
|             if (y >= y_offset && static_cast<uint32_t>(y) < y_offset + this->height_) | ||||
|               buff->draw_pixel_at(x, y, c); | ||||
|           }; | ||||
|           if (!continuous || !has_prev || !prev_b || (abs(y - prev_y) <= thick)) { | ||||
|   | ||||
| @@ -116,7 +116,7 @@ void GraphicalDisplayMenu::draw_menu_internal_(display::Display *display, const | ||||
|   int number_items_fit_to_screen = 0; | ||||
|   const int max_item_index = this->displayed_item_->items_size() - 1; | ||||
|  | ||||
|   for (size_t i = 0; i <= max_item_index; i++) { | ||||
|   for (size_t i = 0; max_item_index >= 0 && i <= static_cast<size_t>(max_item_index); i++) { | ||||
|     const auto *item = this->displayed_item_->get_item(i); | ||||
|     const bool selected = i == this->cursor_index_; | ||||
|     const display::Rect item_dimensions = this->measure_item(display, item, bounds, selected); | ||||
| @@ -174,7 +174,8 @@ void GraphicalDisplayMenu::draw_menu_internal_(display::Display *display, const | ||||
|  | ||||
|   display->filled_rectangle(bounds->x, bounds->y, max_width, total_height, this->background_color_); | ||||
|   auto y_offset = bounds->y; | ||||
|   for (size_t i = first_item_index; i <= last_item_index; i++) { | ||||
|   for (size_t i = static_cast<size_t>(first_item_index); | ||||
|        last_item_index >= 0 && i <= static_cast<size_t>(last_item_index); i++) { | ||||
|     const auto *item = this->displayed_item_->get_item(i); | ||||
|     const bool selected = i == this->cursor_index_; | ||||
|     display::Rect dimensions = menu_dimensions[i]; | ||||
|   | ||||
| @@ -213,7 +213,7 @@ haier_protocol::HandlerError HonClimate::status_handler_(haier_protocol::FrameTy | ||||
|                this->real_control_packet_size_); | ||||
|         this->status_message_callback_.call((const char *) data, data_size); | ||||
|       } else { | ||||
|         ESP_LOGW(TAG, "Status packet too small: %d (should be >= %d)", data_size, this->real_control_packet_size_); | ||||
|         ESP_LOGW(TAG, "Status packet too small: %zu (should be >= %zu)", data_size, this->real_control_packet_size_); | ||||
|       } | ||||
|       switch (this->protocol_phase_) { | ||||
|         case ProtocolPhases::SENDING_FIRST_STATUS_REQUEST: | ||||
| @@ -827,7 +827,7 @@ haier_protocol::HandlerError HonClimate::process_status_message_(const uint8_t * | ||||
|   size_t expected_size = | ||||
|       2 + this->status_message_header_size_ + this->real_control_packet_size_ + this->real_sensors_packet_size_; | ||||
|   if (size < expected_size) { | ||||
|     ESP_LOGW(TAG, "Unexpected message size %d (expexted >= %d)", size, expected_size); | ||||
|     ESP_LOGW(TAG, "Unexpected message size %u (expexted >= %zu)", size, expected_size); | ||||
|     return haier_protocol::HandlerError::WRONG_MESSAGE_STRUCTURE; | ||||
|   } | ||||
|   uint16_t subtype = (((uint16_t) packet_buffer[0]) << 8) + packet_buffer[1]; | ||||
|   | ||||
| @@ -178,7 +178,7 @@ class HonClimate : public HaierClimateBase { | ||||
|   int extra_control_packet_bytes_{0}; | ||||
|   int extra_sensors_packet_bytes_{4}; | ||||
|   int status_message_header_size_{0}; | ||||
|   int real_control_packet_size_{sizeof(hon_protocol::HaierPacketControl)}; | ||||
|   size_t real_control_packet_size_{sizeof(hon_protocol::HaierPacketControl)}; | ||||
|   int real_sensors_packet_size_{sizeof(hon_protocol::HaierPacketSensors) + 4}; | ||||
|   HonControlMethod control_method_; | ||||
|   std::queue<haier_protocol::HaierMessage> control_messages_queue_; | ||||
|   | ||||
| @@ -7,24 +7,20 @@ namespace hdc1080 { | ||||
|  | ||||
| static const char *const TAG = "hdc1080"; | ||||
|  | ||||
| static const uint8_t HDC1080_ADDRESS = 0x40;  // 0b1000000 from datasheet | ||||
| static const uint8_t HDC1080_CMD_CONFIGURATION = 0x02; | ||||
| static const uint8_t HDC1080_CMD_TEMPERATURE = 0x00; | ||||
| static const uint8_t HDC1080_CMD_HUMIDITY = 0x01; | ||||
|  | ||||
| void HDC1080Component::setup() { | ||||
|   const uint8_t data[2] = { | ||||
|       0b00000000,  // resolution 14bit for both humidity and temperature | ||||
|       0b00000000   // reserved | ||||
|   }; | ||||
|   const uint8_t config[2] = {0x00, 0x00};  // resolution 14bit for both humidity and temperature | ||||
|  | ||||
|   if (!this->write_bytes(HDC1080_CMD_CONFIGURATION, data, 2)) { | ||||
|     // as instruction is same as powerup defaults (for now), interpret as warning if this fails | ||||
|     ESP_LOGW(TAG, "HDC1080 initial config instruction error"); | ||||
|     this->status_set_warning(); | ||||
|   // if configuration fails - there is a problem | ||||
|   if (this->write_register(HDC1080_CMD_CONFIGURATION, config, 2) != i2c::ERROR_OK) { | ||||
|     this->mark_failed(); | ||||
|     return; | ||||
|   } | ||||
| } | ||||
|  | ||||
| void HDC1080Component::dump_config() { | ||||
|   ESP_LOGCONFIG(TAG, "HDC1080:"); | ||||
|   LOG_I2C_DEVICE(this); | ||||
| @@ -35,39 +31,51 @@ void HDC1080Component::dump_config() { | ||||
|   LOG_SENSOR("  ", "Temperature", this->temperature_); | ||||
|   LOG_SENSOR("  ", "Humidity", this->humidity_); | ||||
| } | ||||
|  | ||||
| void HDC1080Component::update() { | ||||
|   uint16_t raw_temp; | ||||
|   // regardless of what sensor/s are defined in yaml configuration | ||||
|   // the hdc1080 setup configuration used, requires both temperature and humidity to be read | ||||
|  | ||||
|   this->status_clear_warning(); | ||||
|  | ||||
|   if (this->write(&HDC1080_CMD_TEMPERATURE, 1) != i2c::ERROR_OK) { | ||||
|     this->status_set_warning(); | ||||
|     return; | ||||
|   } | ||||
|   delay(20); | ||||
|   if (this->read(reinterpret_cast<uint8_t *>(&raw_temp), 2) != i2c::ERROR_OK) { | ||||
|  | ||||
|   this->set_timeout(20, [this]() { | ||||
|     uint16_t raw_temperature; | ||||
|     if (this->read(reinterpret_cast<uint8_t *>(&raw_temperature), 2) != i2c::ERROR_OK) { | ||||
|       this->status_set_warning(); | ||||
|       return; | ||||
|     } | ||||
|   raw_temp = i2c::i2ctohs(raw_temp); | ||||
|   float temp = raw_temp * 0.0025177f - 40.0f;  // raw * 2^-16 * 165 - 40 | ||||
|   this->temperature_->publish_state(temp); | ||||
|  | ||||
|   uint16_t raw_humidity; | ||||
|     if (this->temperature_ != nullptr) { | ||||
|       raw_temperature = i2c::i2ctohs(raw_temperature); | ||||
|       float temperature = raw_temperature * 0.0025177f - 40.0f;  // raw * 2^-16 * 165 - 40 | ||||
|       this->temperature_->publish_state(temperature); | ||||
|     } | ||||
|  | ||||
|     if (this->write(&HDC1080_CMD_HUMIDITY, 1) != i2c::ERROR_OK) { | ||||
|       this->status_set_warning(); | ||||
|       return; | ||||
|     } | ||||
|   delay(20); | ||||
|  | ||||
|     this->set_timeout(20, [this]() { | ||||
|       uint16_t raw_humidity; | ||||
|       if (this->read(reinterpret_cast<uint8_t *>(&raw_humidity), 2) != i2c::ERROR_OK) { | ||||
|         this->status_set_warning(); | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       if (this->humidity_ != nullptr) { | ||||
|         raw_humidity = i2c::i2ctohs(raw_humidity); | ||||
|         float humidity = raw_humidity * 0.001525879f;  // raw * 2^-16 * 100 | ||||
|         this->humidity_->publish_state(humidity); | ||||
|  | ||||
|   ESP_LOGD(TAG, "Got temperature=%.1f°C humidity=%.1f%%", temp, humidity); | ||||
|   this->status_clear_warning(); | ||||
|       } | ||||
| float HDC1080Component::get_setup_priority() const { return setup_priority::DATA; } | ||||
|     }); | ||||
|   }); | ||||
| } | ||||
|  | ||||
| }  // namespace hdc1080 | ||||
| }  // namespace esphome | ||||
|   | ||||
| @@ -12,13 +12,11 @@ class HDC1080Component : public PollingComponent, public i2c::I2CDevice { | ||||
|   void set_temperature(sensor::Sensor *temperature) { temperature_ = temperature; } | ||||
|   void set_humidity(sensor::Sensor *humidity) { humidity_ = humidity; } | ||||
|  | ||||
|   /// Setup the sensor and check for connection. | ||||
|   void setup() override; | ||||
|   void dump_config() override; | ||||
|   /// Retrieve the latest sensor values. This operation takes approximately 16ms. | ||||
|   void update() override; | ||||
|  | ||||
|   float get_setup_priority() const override; | ||||
|   float get_setup_priority() const override { return setup_priority::DATA; } | ||||
|  | ||||
|  protected: | ||||
|   sensor::Sensor *temperature_{nullptr}; | ||||
|   | ||||
| @@ -377,7 +377,7 @@ void I2SAudioSpeaker::speaker_task(void *params) { | ||||
|             this_speaker->current_stream_info_.get_bits_per_sample() <= 16) { | ||||
|           size_t len = bytes_read / sizeof(int16_t); | ||||
|           int16_t *tmp_buf = (int16_t *) new_data; | ||||
|           for (int i = 0; i < len; i += 2) { | ||||
|           for (size_t i = 0; i < len; i += 2) { | ||||
|             int16_t tmp = tmp_buf[i]; | ||||
|             tmp_buf[i] = tmp_buf[i + 1]; | ||||
|             tmp_buf[i + 1] = tmp; | ||||
|   | ||||
| @@ -325,7 +325,7 @@ void ILI9XXXDisplay::draw_pixels_at(int x_start, int y_start, int w, int h, cons | ||||
|       // we could deal here with a non-zero y_offset, but if x_offset is zero, y_offset probably will be so don't bother | ||||
|       this->write_array(ptr, w * h * 2); | ||||
|     } else { | ||||
|       for (size_t y = 0; y != h; y++) { | ||||
|       for (size_t y = 0; y != static_cast<size_t>(h); y++) { | ||||
|         this->write_array(ptr + (y + y_offset) * stride + x_offset, w * 2); | ||||
|       } | ||||
|     } | ||||
| @@ -349,7 +349,7 @@ void ILI9XXXDisplay::draw_pixels_at(int x_start, int y_start, int w, int h, cons | ||||
|         App.feed_wdt(); | ||||
|       } | ||||
|       // end of line? Skip to the next. | ||||
|       if (++pixel == w) { | ||||
|       if (++pixel == static_cast<size_t>(w)) { | ||||
|         pixel = 0; | ||||
|         ptr += (x_pad + x_offset) * 2; | ||||
|       } | ||||
|   | ||||
| @@ -19,15 +19,19 @@ std::string build_json(const json_build_t &f) { | ||||
|  | ||||
| bool parse_json(const std::string &data, const json_parse_t &f) { | ||||
|   // NOLINTBEGIN(clang-analyzer-cplusplus.NewDeleteLeaks) false positive with ArduinoJson | ||||
|   JsonDocument doc = parse_json(data); | ||||
|   JsonDocument doc = parse_json(reinterpret_cast<const uint8_t *>(data.c_str()), data.size()); | ||||
|   if (doc.overflowed() || doc.isNull()) | ||||
|     return false; | ||||
|   return f(doc.as<JsonObject>()); | ||||
|   // NOLINTEND(clang-analyzer-cplusplus.NewDeleteLeaks) | ||||
| } | ||||
|  | ||||
| JsonDocument parse_json(const std::string &data) { | ||||
| JsonDocument parse_json(const uint8_t *data, size_t len) { | ||||
|   // NOLINTBEGIN(clang-analyzer-cplusplus.NewDeleteLeaks) false positive with ArduinoJson | ||||
|   if (data == nullptr || len == 0) { | ||||
|     ESP_LOGE(TAG, "No data to parse"); | ||||
|     return JsonObject();  // return unbound object | ||||
|   } | ||||
| #ifdef USE_PSRAM | ||||
|   auto doc_allocator = SpiRamAllocator(); | ||||
|   JsonDocument json_document(&doc_allocator); | ||||
| @@ -38,7 +42,7 @@ JsonDocument parse_json(const std::string &data) { | ||||
|     ESP_LOGE(TAG, "Could not allocate memory for JSON document!"); | ||||
|     return JsonObject();  // return unbound object | ||||
|   } | ||||
|   DeserializationError err = deserializeJson(json_document, data); | ||||
|   DeserializationError err = deserializeJson(json_document, data, len); | ||||
|  | ||||
|   if (err == DeserializationError::Ok) { | ||||
|     return json_document; | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
|  | ||||
| #include <vector> | ||||
|  | ||||
| #include "esphome/core/defines.h" | ||||
| #include "esphome/core/helpers.h" | ||||
|  | ||||
| #define ARDUINOJSON_ENABLE_STD_STRING 1  // NOLINT | ||||
| @@ -49,8 +50,13 @@ std::string build_json(const json_build_t &f); | ||||
|  | ||||
| /// Parse a JSON string and run the provided json parse function if it's valid. | ||||
| bool parse_json(const std::string &data, const json_parse_t &f); | ||||
|  | ||||
| /// Parse a JSON string and return the root JsonDocument (or an unbound object on error) | ||||
| JsonDocument parse_json(const std::string &data); | ||||
| JsonDocument parse_json(const uint8_t *data, size_t len); | ||||
| /// Parse a JSON string and return the root JsonDocument (or an unbound object on error) | ||||
| inline JsonDocument parse_json(const std::string &data) { | ||||
|   return parse_json(reinterpret_cast<const uint8_t *>(data.c_str()), data.size()); | ||||
| } | ||||
|  | ||||
| /// Builder class for creating JSON documents without lambdas | ||||
| class JsonBuilder { | ||||
|   | ||||
| @@ -22,7 +22,7 @@ void KamstrupKMPComponent::dump_config() { | ||||
|   LOG_SENSOR("  ", "Flow", this->flow_sensor_); | ||||
|   LOG_SENSOR("  ", "Volume", this->volume_sensor_); | ||||
|  | ||||
|   for (int i = 0; i < this->custom_sensors_.size(); i++) { | ||||
|   for (size_t i = 0; i < this->custom_sensors_.size(); i++) { | ||||
|     LOG_SENSOR("  ", "Custom Sensor", this->custom_sensors_[i]); | ||||
|     ESP_LOGCONFIG(TAG, "    Command: 0x%04X", this->custom_commands_[i]); | ||||
|   } | ||||
| @@ -268,7 +268,7 @@ void KamstrupKMPComponent::set_sensor_value_(uint16_t command, float value, uint | ||||
|   } | ||||
|  | ||||
|   // Custom sensors | ||||
|   for (int i = 0; i < this->custom_commands_.size(); i++) { | ||||
|   for (size_t i = 0; i < this->custom_commands_.size(); i++) { | ||||
|     if (command == this->custom_commands_[i]) { | ||||
|       this->custom_sensors_[i]->publish_state(value); | ||||
|     } | ||||
|   | ||||
| @@ -13,8 +13,8 @@ class KeyCollector : public Component { | ||||
|   void loop() override; | ||||
|   void dump_config() override; | ||||
|   void set_provider(key_provider::KeyProvider *provider); | ||||
|   void set_min_length(int min_length) { this->min_length_ = min_length; }; | ||||
|   void set_max_length(int max_length) { this->max_length_ = max_length; }; | ||||
|   void set_min_length(uint32_t min_length) { this->min_length_ = min_length; }; | ||||
|   void set_max_length(uint32_t max_length) { this->max_length_ = max_length; }; | ||||
|   void set_start_keys(std::string start_keys) { this->start_keys_ = std::move(start_keys); }; | ||||
|   void set_end_keys(std::string end_keys) { this->end_keys_ = std::move(end_keys); }; | ||||
|   void set_end_key_required(bool end_key_required) { this->end_key_required_ = end_key_required; }; | ||||
| @@ -33,8 +33,8 @@ class KeyCollector : public Component { | ||||
|  protected: | ||||
|   void key_pressed_(uint8_t key); | ||||
|  | ||||
|   int min_length_{0}; | ||||
|   int max_length_{0}; | ||||
|   uint32_t min_length_{0}; | ||||
|   uint32_t max_length_{0}; | ||||
|   std::string start_keys_; | ||||
|   std::string end_keys_; | ||||
|   bool end_key_required_{false}; | ||||
|   | ||||
							
								
								
									
										0
									
								
								esphome/components/lm75b/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								esphome/components/lm75b/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										39
									
								
								esphome/components/lm75b/lm75b.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								esphome/components/lm75b/lm75b.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| #include "lm75b.h" | ||||
| #include "esphome/core/log.h" | ||||
| #include "esphome/core/hal.h" | ||||
|  | ||||
| namespace esphome { | ||||
| namespace lm75b { | ||||
|  | ||||
| static const char *const TAG = "lm75b"; | ||||
|  | ||||
| void LM75BComponent::dump_config() { | ||||
|   ESP_LOGCONFIG(TAG, "LM75B:"); | ||||
|   LOG_I2C_DEVICE(this); | ||||
|   if (this->is_failed()) { | ||||
|     ESP_LOGE(TAG, "Setting up LM75B failed!"); | ||||
|   } | ||||
|   LOG_UPDATE_INTERVAL(this); | ||||
|   LOG_SENSOR("  ", "Temperature", this); | ||||
| } | ||||
|  | ||||
| void LM75BComponent::update() { | ||||
|   // Create a temporary buffer | ||||
|   uint8_t buff[2]; | ||||
|   if (this->read_register(LM75B_REG_TEMPERATURE, buff, 2) != i2c::ERROR_OK) { | ||||
|     this->status_set_warning(); | ||||
|     return; | ||||
|   } | ||||
|   // Obtain combined 16-bit value | ||||
|   int16_t raw_temperature = (buff[0] << 8) | buff[1]; | ||||
|   // Read the 11-bit raw temperature value | ||||
|   raw_temperature >>= 5; | ||||
|   // Publish the temperature in °C | ||||
|   this->publish_state(raw_temperature * 0.125); | ||||
|   if (this->status_has_warning()) { | ||||
|     this->status_clear_warning(); | ||||
|   } | ||||
| } | ||||
|  | ||||
| }  // namespace lm75b | ||||
| }  // namespace esphome | ||||
							
								
								
									
										19
									
								
								esphome/components/lm75b/lm75b.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								esphome/components/lm75b/lm75b.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "esphome/core/component.h" | ||||
| #include "esphome/components/sensor/sensor.h" | ||||
| #include "esphome/components/i2c/i2c.h" | ||||
|  | ||||
| namespace esphome { | ||||
| namespace lm75b { | ||||
|  | ||||
| static const uint8_t LM75B_REG_TEMPERATURE = 0x00; | ||||
|  | ||||
| class LM75BComponent : public PollingComponent, public i2c::I2CDevice, public sensor::Sensor { | ||||
|  public: | ||||
|   void dump_config() override; | ||||
|   void update() override; | ||||
| }; | ||||
|  | ||||
| }  // namespace lm75b | ||||
| }  // namespace esphome | ||||
							
								
								
									
										34
									
								
								esphome/components/lm75b/sensor.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								esphome/components/lm75b/sensor.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| import esphome.codegen as cg | ||||
| from esphome.components import i2c, sensor | ||||
| import esphome.config_validation as cv | ||||
| from esphome.const import ( | ||||
|     DEVICE_CLASS_TEMPERATURE, | ||||
|     STATE_CLASS_MEASUREMENT, | ||||
|     UNIT_CELSIUS, | ||||
| ) | ||||
|  | ||||
| CODEOWNERS = ["@beormund"] | ||||
| DEPENDENCIES = ["i2c"] | ||||
|  | ||||
| lm75b_ns = cg.esphome_ns.namespace("lm75b") | ||||
| LM75BComponent = lm75b_ns.class_( | ||||
|     "LM75BComponent", cg.PollingComponent, i2c.I2CDevice, sensor.Sensor | ||||
| ) | ||||
|  | ||||
| CONFIG_SCHEMA = ( | ||||
|     sensor.sensor_schema( | ||||
|         LM75BComponent, | ||||
|         unit_of_measurement=UNIT_CELSIUS, | ||||
|         accuracy_decimals=3, | ||||
|         device_class=DEVICE_CLASS_TEMPERATURE, | ||||
|         state_class=STATE_CLASS_MEASUREMENT, | ||||
|     ) | ||||
|     .extend(cv.polling_component_schema("60s")) | ||||
|     .extend(i2c.i2c_device_schema(0x48)) | ||||
| ) | ||||
|  | ||||
|  | ||||
| async def to_code(config): | ||||
|     var = await sensor.new_sensor(config) | ||||
|     await cg.register_component(var, config) | ||||
|     await i2c.register_i2c_device(var, config) | ||||
| @@ -5,7 +5,7 @@ | ||||
| #include "esphome/core/helpers.h" | ||||
| #include "esphome/core/log.h" | ||||
| #include "esphome/core/preferences.h" | ||||
| #include <set> | ||||
| #include <initializer_list> | ||||
|  | ||||
| namespace esphome { | ||||
| namespace lock { | ||||
| @@ -44,16 +44,22 @@ class LockTraits { | ||||
|   bool get_assumed_state() const { return this->assumed_state_; } | ||||
|   void set_assumed_state(bool assumed_state) { this->assumed_state_ = assumed_state; } | ||||
|  | ||||
|   bool supports_state(LockState state) const { return supported_states_.count(state); } | ||||
|   std::set<LockState> get_supported_states() const { return supported_states_; } | ||||
|   void set_supported_states(std::set<LockState> states) { supported_states_ = std::move(states); } | ||||
|   void add_supported_state(LockState state) { supported_states_.insert(state); } | ||||
|   bool supports_state(LockState state) const { return supported_states_mask_ & (1 << state); } | ||||
|   void set_supported_states(std::initializer_list<LockState> states) { | ||||
|     supported_states_mask_ = 0; | ||||
|     for (auto state : states) { | ||||
|       supported_states_mask_ |= (1 << state); | ||||
|     } | ||||
|   } | ||||
|   uint8_t get_supported_states_mask() const { return supported_states_mask_; } | ||||
|   void set_supported_states_mask(uint8_t mask) { supported_states_mask_ = mask; } | ||||
|   void add_supported_state(LockState state) { supported_states_mask_ |= (1 << state); } | ||||
|  | ||||
|  protected: | ||||
|   bool supports_open_{false}; | ||||
|   bool requires_code_{false}; | ||||
|   bool assumed_state_{false}; | ||||
|   std::set<LockState> supported_states_ = {LOCK_STATE_NONE, LOCK_STATE_LOCKED, LOCK_STATE_UNLOCKED}; | ||||
|   uint8_t supported_states_mask_{(1 << LOCK_STATE_NONE) | (1 << LOCK_STATE_LOCKED) | (1 << LOCK_STATE_UNLOCKED)}; | ||||
| }; | ||||
|  | ||||
| /** This class is used to encode all control actions on a lock device. | ||||
|   | ||||
| @@ -95,6 +95,7 @@ DEFAULT = "DEFAULT" | ||||
|  | ||||
| CONF_INITIAL_LEVEL = "initial_level" | ||||
| CONF_LOGGER_ID = "logger_id" | ||||
| CONF_RUNTIME_TAG_LEVELS = "runtime_tag_levels" | ||||
| CONF_TASK_LOG_BUFFER_SIZE = "task_log_buffer_size" | ||||
|  | ||||
| UART_SELECTION_ESP32 = { | ||||
| @@ -249,6 +250,7 @@ CONFIG_SCHEMA = cv.All( | ||||
|                 } | ||||
|             ), | ||||
|             cv.Optional(CONF_INITIAL_LEVEL): is_log_level, | ||||
|             cv.Optional(CONF_RUNTIME_TAG_LEVELS, default=False): cv.boolean, | ||||
|             cv.Optional(CONF_ON_MESSAGE): automation.validate_automation( | ||||
|                 { | ||||
|                     cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(LoggerMessageTrigger), | ||||
| @@ -291,7 +293,11 @@ async def to_code(config): | ||||
|         ) | ||||
|     cg.add(log.pre_setup()) | ||||
|  | ||||
|     for tag, log_level in config[CONF_LOGS].items(): | ||||
|     # Enable runtime tag levels if logs are configured or explicitly enabled | ||||
|     logs_config = config[CONF_LOGS] | ||||
|     if logs_config or config[CONF_RUNTIME_TAG_LEVELS]: | ||||
|         cg.add_define("USE_LOGGER_RUNTIME_TAG_LEVELS") | ||||
|         for tag, log_level in logs_config.items(): | ||||
|             cg.add(log.set_log_level(tag, LOG_LEVELS[log_level])) | ||||
|  | ||||
|     cg.add_define("USE_LOGGER") | ||||
| @@ -443,6 +449,7 @@ async def logger_set_level_to_code(config, action_id, template_arg, args): | ||||
|     level = LOG_LEVELS[config[CONF_LEVEL]] | ||||
|     logger = await cg.get_variable(config[CONF_LOGGER_ID]) | ||||
|     if tag := config.get(CONF_TAG): | ||||
|         cg.add_define("USE_LOGGER_RUNTIME_TAG_LEVELS") | ||||
|         text = str(cg.statement(logger.set_log_level(tag, level))) | ||||
|     else: | ||||
|         text = str(cg.statement(logger.set_log_level(level))) | ||||
|   | ||||
| @@ -148,9 +148,11 @@ void Logger::log_vprintf_(uint8_t level, const char *tag, int line, const __Flas | ||||
| #endif  // USE_STORE_LOG_STR_IN_FLASH | ||||
|  | ||||
| inline uint8_t Logger::level_for(const char *tag) { | ||||
| #ifdef USE_LOGGER_RUNTIME_TAG_LEVELS | ||||
|   auto it = this->log_levels_.find(tag); | ||||
|   if (it != this->log_levels_.end()) | ||||
|     return it->second; | ||||
| #endif | ||||
|   return this->current_level_; | ||||
| } | ||||
|  | ||||
| @@ -220,7 +222,9 @@ void Logger::process_messages_() { | ||||
| } | ||||
|  | ||||
| void Logger::set_baud_rate(uint32_t baud_rate) { this->baud_rate_ = baud_rate; } | ||||
| void Logger::set_log_level(const std::string &tag, uint8_t log_level) { this->log_levels_[tag] = log_level; } | ||||
| #ifdef USE_LOGGER_RUNTIME_TAG_LEVELS | ||||
| void Logger::set_log_level(const char *tag, uint8_t log_level) { this->log_levels_[tag] = log_level; } | ||||
| #endif | ||||
|  | ||||
| #if defined(USE_ESP32) || defined(USE_ESP8266) || defined(USE_RP2040) || defined(USE_LIBRETINY) || defined(USE_ZEPHYR) | ||||
| UARTSelection Logger::get_uart() const { return this->uart_; } | ||||
| @@ -271,9 +275,11 @@ void Logger::dump_config() { | ||||
|   } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_LOGGER_RUNTIME_TAG_LEVELS | ||||
|   for (auto &it : this->log_levels_) { | ||||
|     ESP_LOGCONFIG(TAG, "  Level for '%s': %s", it.first.c_str(), LOG_STR_ARG(LOG_LEVELS[it.second])); | ||||
|     ESP_LOGCONFIG(TAG, "  Level for '%s': %s", it.first, LOG_STR_ARG(LOG_LEVELS[it.second])); | ||||
|   } | ||||
| #endif | ||||
| } | ||||
|  | ||||
| void Logger::set_log_level(uint8_t level) { | ||||
|   | ||||
| @@ -36,29 +36,38 @@ struct device; | ||||
|  | ||||
| namespace esphome::logger { | ||||
|  | ||||
| // Color and letter constants for log levels | ||||
| static const char *const LOG_LEVEL_COLORS[] = { | ||||
|     "",                                            // NONE | ||||
|     ESPHOME_LOG_BOLD(ESPHOME_LOG_COLOR_RED),       // ERROR | ||||
|     ESPHOME_LOG_COLOR(ESPHOME_LOG_COLOR_YELLOW),   // WARNING | ||||
|     ESPHOME_LOG_COLOR(ESPHOME_LOG_COLOR_GREEN),    // INFO | ||||
|     ESPHOME_LOG_COLOR(ESPHOME_LOG_COLOR_MAGENTA),  // CONFIG | ||||
|     ESPHOME_LOG_COLOR(ESPHOME_LOG_COLOR_CYAN),     // DEBUG | ||||
|     ESPHOME_LOG_COLOR(ESPHOME_LOG_COLOR_GRAY),     // VERBOSE | ||||
|     ESPHOME_LOG_COLOR(ESPHOME_LOG_COLOR_WHITE),    // VERY_VERBOSE | ||||
| #ifdef USE_LOGGER_RUNTIME_TAG_LEVELS | ||||
| // Comparison function for const char* keys in log_levels_ map | ||||
| struct CStrCompare { | ||||
|   bool operator()(const char *a, const char *b) const { return strcmp(a, b) < 0; } | ||||
| }; | ||||
| #endif | ||||
|  | ||||
| // ANSI color code last digit (30-38 range, store only last digit to save RAM) | ||||
| static constexpr char LOG_LEVEL_COLOR_DIGIT[] = { | ||||
|     '\0',  // NONE | ||||
|     '1',   // ERROR (31 = red) | ||||
|     '3',   // WARNING (33 = yellow) | ||||
|     '2',   // INFO (32 = green) | ||||
|     '5',   // CONFIG (35 = magenta) | ||||
|     '6',   // DEBUG (36 = cyan) | ||||
|     '7',   // VERBOSE (37 = gray) | ||||
|     '8',   // VERY_VERBOSE (38 = white) | ||||
| }; | ||||
|  | ||||
| static const char *const LOG_LEVEL_LETTERS[] = { | ||||
|     "",    // NONE | ||||
|     "E",   // ERROR | ||||
|     "W",   // WARNING | ||||
|     "I",   // INFO | ||||
|     "C",   // CONFIG | ||||
|     "D",   // DEBUG | ||||
|     "V",   // VERBOSE | ||||
|     "VV",  // VERY_VERBOSE | ||||
| static constexpr char LOG_LEVEL_LETTER_CHARS[] = { | ||||
|     '\0',  // NONE | ||||
|     'E',   // ERROR | ||||
|     'W',   // WARNING | ||||
|     'I',   // INFO | ||||
|     'C',   // CONFIG | ||||
|     'D',   // DEBUG | ||||
|     'V',   // VERBOSE (VERY_VERBOSE uses two 'V's) | ||||
| }; | ||||
|  | ||||
| // Maximum header size: 35 bytes fixed + 32 bytes tag + 16 bytes thread name = 83 bytes (45 byte safety margin) | ||||
| static constexpr uint16_t MAX_HEADER_SIZE = 128; | ||||
|  | ||||
| #if defined(USE_ESP32) || defined(USE_ESP8266) || defined(USE_RP2040) || defined(USE_LIBRETINY) || defined(USE_ZEPHYR) | ||||
| /** Enum for logging UART selection | ||||
|  * | ||||
| @@ -131,8 +140,10 @@ class Logger : public Component { | ||||
|  | ||||
|   /// Set the default log level for this logger. | ||||
|   void set_log_level(uint8_t level); | ||||
| #ifdef USE_LOGGER_RUNTIME_TAG_LEVELS | ||||
|   /// Set the log level of the specified tag. | ||||
|   void set_log_level(const std::string &tag, uint8_t log_level); | ||||
|   void set_log_level(const char *tag, uint8_t log_level); | ||||
| #endif | ||||
|   uint8_t get_log_level() { return this->current_level_; } | ||||
|  | ||||
|   // ========== INTERNAL METHODS ========== | ||||
| @@ -215,14 +226,6 @@ class Logger : public Component { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // Format string to explicit buffer with varargs | ||||
|   inline void printf_to_buffer_(char *buffer, uint16_t *buffer_at, uint16_t buffer_size, const char *format, ...) { | ||||
|     va_list arg; | ||||
|     va_start(arg, format); | ||||
|     this->format_body_to_buffer_(buffer, buffer_at, buffer_size, format, arg); | ||||
|     va_end(arg); | ||||
|   } | ||||
|  | ||||
| #ifndef USE_HOST | ||||
|   const LogString *get_uart_selection_(); | ||||
| #endif | ||||
| @@ -248,7 +251,9 @@ class Logger : public Component { | ||||
| #endif | ||||
|  | ||||
|   // Large objects (internally aligned) | ||||
|   std::map<std::string, uint8_t> log_levels_{}; | ||||
| #ifdef USE_LOGGER_RUNTIME_TAG_LEVELS | ||||
|   std::map<const char *, uint8_t, CStrCompare> log_levels_{}; | ||||
| #endif | ||||
|   CallbackManager<void(uint8_t, const char *, const char *, size_t)> log_callback_{}; | ||||
|   CallbackManager<void(uint8_t)> level_callback_{}; | ||||
| #ifdef USE_ESPHOME_TASK_LOG_BUFFER | ||||
| @@ -318,26 +323,76 @@ class Logger : public Component { | ||||
|   } | ||||
| #endif | ||||
|  | ||||
|   static inline void copy_string(char *buffer, uint16_t &pos, const char *str) { | ||||
|     const size_t len = strlen(str); | ||||
|     // Intentionally no null terminator, building larger string | ||||
|     memcpy(buffer + pos, str, len);  // NOLINT(bugprone-not-null-terminated-result) | ||||
|     pos += len; | ||||
|   } | ||||
|  | ||||
|   static inline void write_ansi_color_for_level(char *buffer, uint16_t &pos, uint8_t level) { | ||||
|     if (level == 0) | ||||
|       return; | ||||
|     // Construct ANSI escape sequence: "\033[{bold};3{color}m" | ||||
|     // Example: "\033[1;31m" for ERROR (bold red) | ||||
|     buffer[pos++] = '\033'; | ||||
|     buffer[pos++] = '['; | ||||
|     buffer[pos++] = (level == 1) ? '1' : '0';  // Only ERROR is bold | ||||
|     buffer[pos++] = ';'; | ||||
|     buffer[pos++] = '3'; | ||||
|     buffer[pos++] = LOG_LEVEL_COLOR_DIGIT[level]; | ||||
|     buffer[pos++] = 'm'; | ||||
|   } | ||||
|  | ||||
|   inline void HOT write_header_to_buffer_(uint8_t level, const char *tag, int line, const char *thread_name, | ||||
|                                           char *buffer, uint16_t *buffer_at, uint16_t buffer_size) { | ||||
|     // Format header | ||||
|     // uint8_t level is already bounded 0-255, just ensure it's <= 7 | ||||
|     if (level > 7) | ||||
|       level = 7; | ||||
|     uint16_t pos = *buffer_at; | ||||
|     // Early return if insufficient space - intentionally don't update buffer_at to prevent partial writes | ||||
|     if (pos + MAX_HEADER_SIZE > buffer_size) | ||||
|       return; | ||||
|  | ||||
|     const char *color = esphome::logger::LOG_LEVEL_COLORS[level]; | ||||
|     const char *letter = esphome::logger::LOG_LEVEL_LETTERS[level]; | ||||
|     // Construct: <color>[LEVEL][tag:line]: | ||||
|     write_ansi_color_for_level(buffer, pos, level); | ||||
|     buffer[pos++] = '['; | ||||
|     if (level != 0) { | ||||
|       if (level >= 7) { | ||||
|         buffer[pos++] = 'V';  // VERY_VERBOSE = "VV" | ||||
|         buffer[pos++] = 'V'; | ||||
|       } else { | ||||
|         buffer[pos++] = LOG_LEVEL_LETTER_CHARS[level]; | ||||
|       } | ||||
|     } | ||||
|     buffer[pos++] = ']'; | ||||
|     buffer[pos++] = '['; | ||||
|     copy_string(buffer, pos, tag); | ||||
|     buffer[pos++] = ':'; | ||||
|     // Format line number without modulo operations (passed by value, safe to mutate) | ||||
|     if (line > 999) [[unlikely]] { | ||||
|       int thousands = line / 1000; | ||||
|       buffer[pos++] = '0' + thousands; | ||||
|       line -= thousands * 1000; | ||||
|     } | ||||
|     int hundreds = line / 100; | ||||
|     int remainder = line - hundreds * 100; | ||||
|     int tens = remainder / 10; | ||||
|     buffer[pos++] = '0' + hundreds; | ||||
|     buffer[pos++] = '0' + tens; | ||||
|     buffer[pos++] = '0' + (remainder - tens * 10); | ||||
|     buffer[pos++] = ']'; | ||||
|  | ||||
| #if defined(USE_ESP32) || defined(USE_LIBRETINY) || defined(USE_ZEPHYR) | ||||
|     if (thread_name != nullptr) { | ||||
|       // Non-main task with thread name | ||||
|       this->printf_to_buffer_(buffer, buffer_at, buffer_size, "%s[%s][%s:%03u]%s[%s]%s: ", color, letter, tag, line, | ||||
|                               ESPHOME_LOG_BOLD(ESPHOME_LOG_COLOR_RED), thread_name, color); | ||||
|       return; | ||||
|       write_ansi_color_for_level(buffer, pos, 1);  // Always use bold red for thread name | ||||
|       buffer[pos++] = '['; | ||||
|       copy_string(buffer, pos, thread_name); | ||||
|       buffer[pos++] = ']'; | ||||
|       write_ansi_color_for_level(buffer, pos, level);  // Restore original color | ||||
|     } | ||||
| #endif | ||||
|     // Main task or non ESP32/LibreTiny platform | ||||
|     this->printf_to_buffer_(buffer, buffer_at, buffer_size, "%s[%s][%s:%03u]: ", color, letter, tag, line); | ||||
|  | ||||
|     buffer[pos++] = ':'; | ||||
|     buffer[pos++] = ' '; | ||||
|     *buffer_at = pos; | ||||
|   } | ||||
|  | ||||
|   inline void HOT format_body_to_buffer_(char *buffer, uint16_t *buffer_at, uint16_t buffer_size, const char *format, | ||||
|   | ||||
| @@ -3,11 +3,10 @@ | ||||
| namespace esphome::logger { | ||||
|  | ||||
| void LoggerLevelSelect::publish_state(int level) { | ||||
|   auto value = this->at(level); | ||||
|   if (!value) { | ||||
|   const auto &option = this->at(level_to_index(level)); | ||||
|   if (!option) | ||||
|     return; | ||||
|   } | ||||
|   Select::publish_state(value.value()); | ||||
|   Select::publish_state(option.value()); | ||||
| } | ||||
|  | ||||
| void LoggerLevelSelect::setup() { | ||||
| @@ -16,10 +15,10 @@ void LoggerLevelSelect::setup() { | ||||
| } | ||||
|  | ||||
| void LoggerLevelSelect::control(const std::string &value) { | ||||
|   auto level = this->index_of(value); | ||||
|   if (!level) | ||||
|   const auto index = this->index_of(value); | ||||
|   if (!index) | ||||
|     return; | ||||
|   this->parent_->set_log_level(level.value()); | ||||
|   this->parent_->set_log_level(index_to_level(index.value())); | ||||
| } | ||||
|  | ||||
| }  // namespace esphome::logger | ||||
|   | ||||
| @@ -3,11 +3,18 @@ | ||||
| #include "esphome/components/select/select.h" | ||||
| #include "esphome/core/component.h" | ||||
| #include "esphome/components/logger/logger.h" | ||||
|  | ||||
| namespace esphome::logger { | ||||
| class LoggerLevelSelect : public Component, public select::Select, public Parented<Logger> { | ||||
|  public: | ||||
|   void publish_state(int level); | ||||
|   void setup() override; | ||||
|   void control(const std::string &value) override; | ||||
|  | ||||
|  protected: | ||||
|   // Convert log level to option index (skip CONFIG at level 4) | ||||
|   static uint8_t level_to_index(uint8_t level) { return (level > ESPHOME_LOG_LEVEL_CONFIG) ? level - 1 : level; } | ||||
|   // Convert option index to log level (skip CONFIG at level 4) | ||||
|   static uint8_t index_to_level(uint8_t index) { return (index >= ESPHOME_LOG_LEVEL_CONFIG) ? index + 1 : index; } | ||||
| }; | ||||
| }  // namespace esphome::logger | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| #include "esphome/core/application.h" | ||||
| #include "esphome/core/helpers.h" | ||||
| #include "esphome/core/log.h" | ||||
| #include <limits> | ||||
|  | ||||
| using esphome::i2c::ErrorCode; | ||||
|  | ||||
| @@ -28,30 +29,30 @@ bool operator!=(const GainTimePair &lhs, const GainTimePair &rhs) { | ||||
|  | ||||
| template<typename T, size_t size> T get_next(const T (&array)[size], const T val) { | ||||
|   size_t i = 0; | ||||
|   size_t idx = -1; | ||||
|   while (idx == -1 && i < size) { | ||||
|   size_t idx = std::numeric_limits<size_t>::max(); | ||||
|   while (idx == std::numeric_limits<size_t>::max() && i < size) { | ||||
|     if (array[i] == val) { | ||||
|       idx = i; | ||||
|       break; | ||||
|     } | ||||
|     i++; | ||||
|   } | ||||
|   if (idx == -1 || i + 1 >= size) | ||||
|   if (idx == std::numeric_limits<size_t>::max() || i + 1 >= size) | ||||
|     return val; | ||||
|   return array[i + 1]; | ||||
| } | ||||
|  | ||||
| template<typename T, size_t size> T get_prev(const T (&array)[size], const T val) { | ||||
|   size_t i = size - 1; | ||||
|   size_t idx = -1; | ||||
|   while (idx == -1 && i > 0) { | ||||
|   size_t idx = std::numeric_limits<size_t>::max(); | ||||
|   while (idx == std::numeric_limits<size_t>::max() && i > 0) { | ||||
|     if (array[i] == val) { | ||||
|       idx = i; | ||||
|       break; | ||||
|     } | ||||
|     i--; | ||||
|   } | ||||
|   if (idx == -1 || i == 0) | ||||
|   if (idx == std::numeric_limits<size_t>::max() || i == 0) | ||||
|     return val; | ||||
|   return array[i - 1]; | ||||
| } | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| #include "esphome/core/application.h" | ||||
| #include "esphome/core/helpers.h" | ||||
| #include "esphome/core/log.h" | ||||
| #include <limits> | ||||
|  | ||||
| using esphome::i2c::ErrorCode; | ||||
|  | ||||
| @@ -14,30 +15,30 @@ static const uint8_t MAX_TRIES = 5; | ||||
|  | ||||
| template<typename T, size_t size> T get_next(const T (&array)[size], const T val) { | ||||
|   size_t i = 0; | ||||
|   size_t idx = -1; | ||||
|   while (idx == -1 && i < size) { | ||||
|   size_t idx = std::numeric_limits<size_t>::max(); | ||||
|   while (idx == std::numeric_limits<size_t>::max() && i < size) { | ||||
|     if (array[i] == val) { | ||||
|       idx = i; | ||||
|       break; | ||||
|     } | ||||
|     i++; | ||||
|   } | ||||
|   if (idx == -1 || i + 1 >= size) | ||||
|   if (idx == std::numeric_limits<size_t>::max() || i + 1 >= size) | ||||
|     return val; | ||||
|   return array[i + 1]; | ||||
| } | ||||
|  | ||||
| template<typename T, size_t size> T get_prev(const T (&array)[size], const T val) { | ||||
|   size_t i = size - 1; | ||||
|   size_t idx = -1; | ||||
|   while (idx == -1 && i > 0) { | ||||
|   size_t idx = std::numeric_limits<size_t>::max(); | ||||
|   while (idx == std::numeric_limits<size_t>::max() && i > 0) { | ||||
|     if (array[i] == val) { | ||||
|       idx = i; | ||||
|       break; | ||||
|     } | ||||
|     i--; | ||||
|   } | ||||
|   if (idx == -1 || i == 0) | ||||
|   if (idx == std::numeric_limits<size_t>::max() || i == 0) | ||||
|     return val; | ||||
|   return array[i - 1]; | ||||
| } | ||||
|   | ||||
| @@ -29,9 +29,9 @@ class MatrixKeypad : public key_provider::KeyProvider, public Component { | ||||
|   void set_columns(std::vector<GPIOPin *> pins) { columns_ = std::move(pins); }; | ||||
|   void set_rows(std::vector<GPIOPin *> pins) { rows_ = std::move(pins); }; | ||||
|   void set_keys(std::string keys) { keys_ = std::move(keys); }; | ||||
|   void set_debounce_time(int debounce_time) { debounce_time_ = debounce_time; }; | ||||
|   void set_has_diodes(int has_diodes) { has_diodes_ = has_diodes; }; | ||||
|   void set_has_pulldowns(int has_pulldowns) { has_pulldowns_ = has_pulldowns; }; | ||||
|   void set_debounce_time(uint32_t debounce_time) { debounce_time_ = debounce_time; }; | ||||
|   void set_has_diodes(bool has_diodes) { has_diodes_ = has_diodes; }; | ||||
|   void set_has_pulldowns(bool has_pulldowns) { has_pulldowns_ = has_pulldowns; }; | ||||
|  | ||||
|   void register_listener(MatrixKeypadListener *listener); | ||||
|   void register_key_trigger(MatrixKeyTrigger *trig); | ||||
| @@ -40,7 +40,7 @@ class MatrixKeypad : public key_provider::KeyProvider, public Component { | ||||
|   std::vector<GPIOPin *> rows_; | ||||
|   std::vector<GPIOPin *> columns_; | ||||
|   std::string keys_; | ||||
|   int debounce_time_ = 0; | ||||
|   uint32_t debounce_time_ = 0; | ||||
|   bool has_diodes_{false}; | ||||
|   bool has_pulldowns_{false}; | ||||
|   int pressed_key_ = -1; | ||||
|   | ||||
| @@ -90,7 +90,7 @@ void MAX7219Component::loop() { | ||||
|   } | ||||
|  | ||||
|   if (this->scroll_mode_ == ScrollMode::STOP) { | ||||
|     if (this->stepsleft_ + get_width_internal() == first_line_size + 1) { | ||||
|     if (static_cast<size_t>(this->stepsleft_ + get_width_internal()) == first_line_size + 1) { | ||||
|       if (millis_since_last_scroll < this->scroll_dwell_) { | ||||
|         ESP_LOGVV(TAG, "Dwell time at end of string in case of stop at end. Step %d, since last scroll %d, dwell %d.", | ||||
|                   this->stepsleft_, millis_since_last_scroll, this->scroll_dwell_); | ||||
|   | ||||
| @@ -17,6 +17,11 @@ from esphome.coroutine import CoroPriority | ||||
| CODEOWNERS = ["@esphome/core"] | ||||
| DEPENDENCIES = ["network"] | ||||
|  | ||||
| # Components that create mDNS services at runtime | ||||
| # IMPORTANT: If you add a new component here, you must also update the corresponding | ||||
| # #ifdef blocks in mdns_component.cpp compile_records_() method | ||||
| COMPONENTS_WITH_MDNS_SERVICES = ("api", "prometheus", "web_server") | ||||
|  | ||||
| mdns_ns = cg.esphome_ns.namespace("mdns") | ||||
| MDNSComponent = mdns_ns.class_("MDNSComponent", cg.Component) | ||||
| MDNSTXTRecord = mdns_ns.struct("MDNSTXTRecord") | ||||
| @@ -91,12 +96,20 @@ async def to_code(config): | ||||
|  | ||||
|     cg.add_define("USE_MDNS") | ||||
|  | ||||
|     var = cg.new_Pvariable(config[CONF_ID]) | ||||
|     await cg.register_component(var, config) | ||||
|     # Calculate compile-time service count | ||||
|     service_count = sum( | ||||
|         1 for key in COMPONENTS_WITH_MDNS_SERVICES if key in CORE.config | ||||
|     ) + len(config[CONF_SERVICES]) | ||||
|  | ||||
|     if config[CONF_SERVICES]: | ||||
|         cg.add_define("USE_MDNS_EXTRA_SERVICES") | ||||
|  | ||||
|     # Ensure at least 1 service (fallback service) | ||||
|     cg.add_define("MDNS_SERVICE_COUNT", max(1, service_count)) | ||||
|  | ||||
|     var = cg.new_Pvariable(config[CONF_ID]) | ||||
|     await cg.register_component(var, config) | ||||
|  | ||||
|     for service in config[CONF_SERVICES]: | ||||
|         txt = [ | ||||
|             cg.StructInitializer( | ||||
|   | ||||
| @@ -74,32 +74,12 @@ MDNS_STATIC_CONST_CHAR(NETWORK_THREAD, "thread"); | ||||
| void MDNSComponent::compile_records_() { | ||||
|   this->hostname_ = App.get_name(); | ||||
|  | ||||
|   // Calculate exact capacity needed for services vector | ||||
|   size_t services_count = 0; | ||||
| #ifdef USE_API | ||||
|   if (api::global_api_server != nullptr) { | ||||
|     services_count++; | ||||
|   } | ||||
| #endif | ||||
| #ifdef USE_PROMETHEUS | ||||
|   services_count++; | ||||
| #endif | ||||
| #ifdef USE_WEBSERVER | ||||
|   services_count++; | ||||
| #endif | ||||
| #ifdef USE_MDNS_EXTRA_SERVICES | ||||
|   services_count += this->services_extra_.size(); | ||||
| #endif | ||||
|   // Reserve for fallback service if needed | ||||
|   if (services_count == 0) { | ||||
|     services_count = 1; | ||||
|   } | ||||
|   this->services_.reserve(services_count); | ||||
|   // IMPORTANT: The #ifdef blocks below must match COMPONENTS_WITH_MDNS_SERVICES | ||||
|   // in mdns/__init__.py. If you add a new service here, update both locations. | ||||
|  | ||||
| #ifdef USE_API | ||||
|   if (api::global_api_server != nullptr) { | ||||
|     this->services_.emplace_back(); | ||||
|     auto &service = this->services_.back(); | ||||
|     auto &service = this->services_.emplace_next(); | ||||
|     service.service_type = MDNS_STR(SERVICE_ESPHOMELIB); | ||||
|     service.proto = MDNS_STR(SERVICE_TCP); | ||||
|     service.port = api::global_api_server->get_port(); | ||||
| @@ -178,30 +158,23 @@ void MDNSComponent::compile_records_() { | ||||
| #endif  // USE_API | ||||
|  | ||||
| #ifdef USE_PROMETHEUS | ||||
|   this->services_.emplace_back(); | ||||
|   auto &prom_service = this->services_.back(); | ||||
|   auto &prom_service = this->services_.emplace_next(); | ||||
|   prom_service.service_type = MDNS_STR(SERVICE_PROMETHEUS); | ||||
|   prom_service.proto = MDNS_STR(SERVICE_TCP); | ||||
|   prom_service.port = USE_WEBSERVER_PORT; | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_WEBSERVER | ||||
|   this->services_.emplace_back(); | ||||
|   auto &web_service = this->services_.back(); | ||||
|   auto &web_service = this->services_.emplace_next(); | ||||
|   web_service.service_type = MDNS_STR(SERVICE_HTTP); | ||||
|   web_service.proto = MDNS_STR(SERVICE_TCP); | ||||
|   web_service.port = USE_WEBSERVER_PORT; | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_MDNS_EXTRA_SERVICES | ||||
|   this->services_.insert(this->services_.end(), this->services_extra_.begin(), this->services_extra_.end()); | ||||
| #endif | ||||
|  | ||||
| #if !defined(USE_API) && !defined(USE_PROMETHEUS) && !defined(USE_WEBSERVER) && !defined(USE_MDNS_EXTRA_SERVICES) | ||||
|   // Publish "http" service if not using native API or any other services | ||||
|   // This is just to have *some* mDNS service so that .local resolution works | ||||
|   this->services_.emplace_back(); | ||||
|   auto &fallback_service = this->services_.back(); | ||||
|   auto &fallback_service = this->services_.emplace_next(); | ||||
|   fallback_service.service_type = "_http"; | ||||
|   fallback_service.proto = "_tcp"; | ||||
|   fallback_service.port = USE_WEBSERVER_PORT; | ||||
| @@ -214,7 +187,7 @@ void MDNSComponent::dump_config() { | ||||
|                 "mDNS:\n" | ||||
|                 "  Hostname: %s", | ||||
|                 this->hostname_.c_str()); | ||||
| #if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERY_VERBOSE | ||||
| #if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE | ||||
|   ESP_LOGV(TAG, "  Services:"); | ||||
|   for (const auto &service : this->services_) { | ||||
|     ESP_LOGV(TAG, "  - %s, %s, %d", service.service_type.c_str(), service.proto.c_str(), | ||||
| @@ -227,8 +200,6 @@ void MDNSComponent::dump_config() { | ||||
| #endif | ||||
| } | ||||
|  | ||||
| std::vector<MDNSService> MDNSComponent::get_services() { return this->services_; } | ||||
|  | ||||
| }  // namespace mdns | ||||
| }  // namespace esphome | ||||
| #endif | ||||
|   | ||||
| @@ -2,13 +2,16 @@ | ||||
| #include "esphome/core/defines.h" | ||||
| #ifdef USE_MDNS | ||||
| #include <string> | ||||
| #include <vector> | ||||
| #include "esphome/core/automation.h" | ||||
| #include "esphome/core/component.h" | ||||
| #include "esphome/core/helpers.h" | ||||
|  | ||||
| namespace esphome { | ||||
| namespace mdns { | ||||
|  | ||||
| // Service count is calculated at compile time by Python codegen | ||||
| // MDNS_SERVICE_COUNT will always be defined | ||||
|  | ||||
| struct MDNSTXTRecord { | ||||
|   std::string key; | ||||
|   TemplatableValue<std::string> value; | ||||
| @@ -36,18 +39,15 @@ class MDNSComponent : public Component { | ||||
|   float get_setup_priority() const override { return setup_priority::AFTER_CONNECTION; } | ||||
|  | ||||
| #ifdef USE_MDNS_EXTRA_SERVICES | ||||
|   void add_extra_service(MDNSService service) { services_extra_.push_back(std::move(service)); } | ||||
|   void add_extra_service(MDNSService service) { this->services_.emplace_next() = std::move(service); } | ||||
| #endif | ||||
|  | ||||
|   std::vector<MDNSService> get_services(); | ||||
|   const StaticVector<MDNSService, MDNS_SERVICE_COUNT> &get_services() const { return this->services_; } | ||||
|  | ||||
|   void on_shutdown() override; | ||||
|  | ||||
|  protected: | ||||
| #ifdef USE_MDNS_EXTRA_SERVICES | ||||
|   std::vector<MDNSService> services_extra_{}; | ||||
| #endif | ||||
|   std::vector<MDNSService> services_{}; | ||||
|   StaticVector<MDNSService, MDNS_SERVICE_COUNT> services_{}; | ||||
|   std::string hostname_; | ||||
|   void compile_records_(); | ||||
| }; | ||||
|   | ||||
| @@ -343,11 +343,7 @@ class DriverChip: | ||||
|             ) | ||||
|             offset_height = native_height - height - offset_height | ||||
|         # Swap default dimensions if swap_xy is set, or if rotation is 90/270 and we are not using a buffer | ||||
|         rotated = not requires_buffer(config) and config.get(CONF_ROTATION, 0) in ( | ||||
|             90, | ||||
|             270, | ||||
|         ) | ||||
|         if transform.get(CONF_SWAP_XY) is True or rotated: | ||||
|         if transform.get(CONF_SWAP_XY) is True: | ||||
|             width, height = height, width | ||||
|             offset_height, offset_width = offset_width, offset_height | ||||
|         return width, height, offset_width, offset_height | ||||
|   | ||||
| @@ -380,25 +380,41 @@ def get_instance(config): | ||||
|         bus_type = BusTypes[bus_type] | ||||
|     buffer_type = cg.uint8 if color_depth == 8 else cg.uint16 | ||||
|     frac = denominator(config) | ||||
|     rotation = DISPLAY_ROTATIONS[ | ||||
|     rotation = ( | ||||
|         0 if model.rotation_as_transform(config) else config.get(CONF_ROTATION, 0) | ||||
|     ] | ||||
|     ) | ||||
|     templateargs = [ | ||||
|         buffer_type, | ||||
|         bufferpixels, | ||||
|         config[CONF_BYTE_ORDER] == "big_endian", | ||||
|         display_pixel_mode, | ||||
|         bus_type, | ||||
|     ] | ||||
|     # If a buffer is required, use MipiSpiBuffer, otherwise use MipiSpi | ||||
|     if requires_buffer(config): | ||||
|         templateargs.extend( | ||||
|             [ | ||||
|                 width, | ||||
|                 height, | ||||
|                 offset_width, | ||||
|                 offset_height, | ||||
|                 DISPLAY_ROTATIONS[rotation], | ||||
|                 frac, | ||||
|             ] | ||||
|         ) | ||||
|         return MipiSpiBuffer, templateargs | ||||
|     # Swap height and width if the display is rotated 90 or 270 degrees in software | ||||
|     if rotation in (90, 270): | ||||
|         width, height = height, width | ||||
|         offset_width, offset_height = offset_height, offset_width | ||||
|     templateargs.extend( | ||||
|         [ | ||||
|             width, | ||||
|             height, | ||||
|             offset_width, | ||||
|             offset_height, | ||||
|         ] | ||||
|     # If a buffer is required, use MipiSpiBuffer, otherwise use MipiSpi | ||||
|     if requires_buffer(config): | ||||
|         templateargs.append(rotation) | ||||
|         templateargs.append(frac) | ||||
|         return MipiSpiBuffer, templateargs | ||||
|     ) | ||||
|     return MipiSpi, templateargs | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -340,7 +340,7 @@ class MipiSpi : public display::Display, | ||||
|         this->write_cmd_addr_data(0, 0, 0, 0, ptr, w * h, 8); | ||||
|       } | ||||
|     } else { | ||||
|       for (size_t y = 0; y != h; y++) { | ||||
|       for (size_t y = 0; y != static_cast<size_t>(h); y++) { | ||||
|         if constexpr (BUS_TYPE == BUS_TYPE_SINGLE || BUS_TYPE == BUS_TYPE_SINGLE_16) { | ||||
|           this->write_array(ptr, w); | ||||
|         } else if constexpr (BUS_TYPE == BUS_TYPE_QUAD) { | ||||
| @@ -372,8 +372,8 @@ class MipiSpi : public display::Display, | ||||
|       uint8_t dbuffer[DISPLAYPIXEL * 48]; | ||||
|       uint8_t *dptr = dbuffer; | ||||
|       auto stride = x_offset + w + x_pad;  // stride in pixels | ||||
|       for (size_t y = 0; y != h; y++) { | ||||
|         for (size_t x = 0; x != w; x++) { | ||||
|       for (size_t y = 0; y != static_cast<size_t>(h); y++) { | ||||
|         for (size_t x = 0; x != static_cast<size_t>(w); x++) { | ||||
|           auto color_val = ptr[y * stride + x]; | ||||
|           if constexpr (DISPLAYPIXEL == PIXEL_MODE_18 && BUFFERPIXEL == PIXEL_MODE_16) { | ||||
|             // 16 to 18 bit conversion | ||||
|   | ||||
| @@ -572,7 +572,7 @@ void MixerSpeaker::audio_mixer_task(void *params) { | ||||
|       } | ||||
|     } else { | ||||
|       // Determine how many frames to mix | ||||
|       for (int i = 0; i < transfer_buffers_with_data.size(); ++i) { | ||||
|       for (size_t i = 0; i < transfer_buffers_with_data.size(); ++i) { | ||||
|         const uint32_t frames_available_in_buffer = | ||||
|             speakers_with_data[i]->get_audio_stream_info().bytes_to_frames(transfer_buffers_with_data[i]->available()); | ||||
|         frames_to_mix = std::min(frames_to_mix, frames_available_in_buffer); | ||||
| @@ -581,7 +581,7 @@ void MixerSpeaker::audio_mixer_task(void *params) { | ||||
|       audio::AudioStreamInfo primary_stream_info = speakers_with_data[0]->get_audio_stream_info(); | ||||
|  | ||||
|       // Mix two streams together | ||||
|       for (int i = 1; i < transfer_buffers_with_data.size(); ++i) { | ||||
|       for (size_t i = 1; i < transfer_buffers_with_data.size(); ++i) { | ||||
|         mix_audio_samples(primary_buffer, primary_stream_info, | ||||
|                           reinterpret_cast<int16_t *>(transfer_buffers_with_data[i]->get_buffer_start()), | ||||
|                           speakers_with_data[i]->get_audio_stream_info(), | ||||
| @@ -596,7 +596,7 @@ void MixerSpeaker::audio_mixer_task(void *params) { | ||||
|       } | ||||
|  | ||||
|       // Update source transfer buffer lengths and add new audio durations to the source speaker pending playbacks | ||||
|       for (int i = 0; i < transfer_buffers_with_data.size(); ++i) { | ||||
|       for (size_t i = 0; i < transfer_buffers_with_data.size(); ++i) { | ||||
|         transfer_buffers_with_data[i]->decrease_buffer_length( | ||||
|             speakers_with_data[i]->get_audio_stream_info().frames_to_bytes(frames_to_mix)); | ||||
|         speakers_with_data[i]->pending_playback_frames_ += frames_to_mix; | ||||
|   | ||||
| @@ -11,15 +11,15 @@ namespace mpr121 { | ||||
| static const char *const TAG = "mpr121"; | ||||
|  | ||||
| void MPR121Component::setup() { | ||||
|   this->disable_loop(); | ||||
|   // soft reset device | ||||
|   this->write_byte(MPR121_SOFTRESET, 0x63); | ||||
|   delay(100);  // NOLINT | ||||
|   this->set_timeout(100, [this]() { | ||||
|     if (!this->write_byte(MPR121_ECR, 0x0)) { | ||||
|       this->error_code_ = COMMUNICATION_FAILED; | ||||
|       this->mark_failed(); | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     // set touch sensitivity for all 12 channels | ||||
|     for (auto *channel : this->channels_) { | ||||
|       channel->setup(); | ||||
| @@ -52,6 +52,8 @@ void MPR121Component::setup() { | ||||
|     this->write_byte(MPR121_ECR, 0x80 | (this->max_touch_channel_ + 1)); | ||||
|  | ||||
|     this->flush_gpio_(); | ||||
|     this->enable_loop(); | ||||
|   }); | ||||
| } | ||||
|  | ||||
| void MPR121Component::set_touch_debounce(uint8_t debounce) { | ||||
| @@ -73,9 +75,6 @@ void MPR121Component::dump_config() { | ||||
|     case COMMUNICATION_FAILED: | ||||
|       ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL); | ||||
|       break; | ||||
|     case WRONG_CHIP_STATE: | ||||
|       ESP_LOGE(TAG, "MPR121 has wrong default value for CONFIG2?"); | ||||
|       break; | ||||
|     case NONE: | ||||
|     default: | ||||
|       break; | ||||
|   | ||||
| @@ -88,7 +88,6 @@ class MPR121Component : public Component, public i2c::I2CDevice { | ||||
|   enum ErrorCode { | ||||
|     NONE = 0, | ||||
|     COMMUNICATION_FAILED, | ||||
|     WRONG_CHIP_STATE, | ||||
|   } error_code_{NONE}; | ||||
|  | ||||
|   bool flush_gpio_(); | ||||
|   | ||||
| @@ -218,7 +218,7 @@ void NAU7802Sensor::dump_config() { | ||||
|  | ||||
| void NAU7802Sensor::write_value_(uint8_t start_reg, size_t size, int32_t value) { | ||||
|   uint8_t data[4]; | ||||
|   for (int i = 0; i < size; i++) { | ||||
|   for (size_t i = 0; i < size; i++) { | ||||
|     data[i] = 0xFF & (value >> (size - 1 - i) * 8); | ||||
|   } | ||||
|   this->write_register(start_reg, data, size); | ||||
| @@ -228,7 +228,7 @@ int32_t NAU7802Sensor::read_value_(uint8_t start_reg, size_t size) { | ||||
|   uint8_t data[4]; | ||||
|   this->read_register(start_reg, data, size); | ||||
|   int32_t result = 0; | ||||
|   for (int i = 0; i < size; i++) { | ||||
|   for (size_t i = 0; i < size; i++) { | ||||
|     result |= data[i] << (size - 1 - i) * 8; | ||||
|   } | ||||
|   // extend sign bit | ||||
|   | ||||
| @@ -77,7 +77,7 @@ bool Nextion::check_connect_() { | ||||
|   this->recv_ret_string_(response, 0, false); | ||||
|   if (!response.empty() && response[0] == 0x1A) { | ||||
|     // Swallow invalid variable name responses that may be caused by the above commands | ||||
|     ESP_LOGD(TAG, "0x1A error ignored (setup)"); | ||||
|     ESP_LOGV(TAG, "0x1A error ignored (setup)"); | ||||
|     return false; | ||||
|   } | ||||
|   if (response.empty() || response.find("comok") == std::string::npos) { | ||||
| @@ -334,7 +334,7 @@ void Nextion::loop() { | ||||
|       this->started_ms_ = App.get_loop_component_start_time(); | ||||
|  | ||||
|     if (this->started_ms_ + this->startup_override_ms_ < App.get_loop_component_start_time()) { | ||||
|       ESP_LOGD(TAG, "Manual ready set"); | ||||
|       ESP_LOGV(TAG, "Manual ready set"); | ||||
|       this->connection_state_.nextion_reports_is_setup_ = true; | ||||
|     } | ||||
|   } | ||||
| @@ -544,7 +544,7 @@ void Nextion::process_nextion_commands_() { | ||||
|         uint8_t page_id = to_process[0]; | ||||
|         uint8_t component_id = to_process[1]; | ||||
|         uint8_t touch_event = to_process[2];  // 0 -> release, 1 -> press | ||||
|         ESP_LOGD(TAG, "Touch %s: page %u comp %u", touch_event ? "PRESS" : "RELEASE", page_id, component_id); | ||||
|         ESP_LOGV(TAG, "Touch %s: page %u comp %u", touch_event ? "PRESS" : "RELEASE", page_id, component_id); | ||||
|         for (auto *touch : this->touch_) { | ||||
|           touch->process_touch(page_id, component_id, touch_event != 0); | ||||
|         } | ||||
| @@ -559,7 +559,7 @@ void Nextion::process_nextion_commands_() { | ||||
|         } | ||||
|  | ||||
|         uint8_t page_id = to_process[0]; | ||||
|         ESP_LOGD(TAG, "New page: %u", page_id); | ||||
|         ESP_LOGV(TAG, "New page: %u", page_id); | ||||
|         this->page_callback_.call(page_id); | ||||
|         break; | ||||
|       } | ||||
| @@ -577,7 +577,7 @@ void Nextion::process_nextion_commands_() { | ||||
|         const uint16_t x = (uint16_t(to_process[0]) << 8) | to_process[1]; | ||||
|         const uint16_t y = (uint16_t(to_process[2]) << 8) | to_process[3]; | ||||
|         const uint8_t touch_event = to_process[4];  // 0 -> release, 1 -> press | ||||
|         ESP_LOGD(TAG, "Touch %s at %u,%u", touch_event ? "PRESS" : "RELEASE", x, y); | ||||
|         ESP_LOGV(TAG, "Touch %s at %u,%u", touch_event ? "PRESS" : "RELEASE", x, y); | ||||
|         break; | ||||
|       } | ||||
|  | ||||
| @@ -676,7 +676,7 @@ void Nextion::process_nextion_commands_() { | ||||
|       } | ||||
|       case 0x88:  // system successful start up | ||||
|       { | ||||
|         ESP_LOGD(TAG, "System start: %zu", to_process_length); | ||||
|         ESP_LOGV(TAG, "System start: %zu", to_process_length); | ||||
|         this->connection_state_.nextion_reports_is_setup_ = true; | ||||
|         break; | ||||
|       } | ||||
| @@ -922,7 +922,7 @@ void Nextion::set_nextion_sensor_state(NextionQueueType queue_type, const std::s | ||||
| } | ||||
|  | ||||
| void Nextion::set_nextion_text_state(const std::string &name, const std::string &state) { | ||||
|   ESP_LOGD(TAG, "State: %s='%s'", name.c_str(), state.c_str()); | ||||
|   ESP_LOGV(TAG, "State: %s='%s'", name.c_str(), state.c_str()); | ||||
|  | ||||
|   for (auto *sensor : this->textsensortype_) { | ||||
|     if (name == sensor->get_variable_name()) { | ||||
| @@ -933,7 +933,7 @@ void Nextion::set_nextion_text_state(const std::string &name, const std::string | ||||
| } | ||||
|  | ||||
| void Nextion::all_components_send_state_(bool force_update) { | ||||
|   ESP_LOGD(TAG, "Send states"); | ||||
|   ESP_LOGV(TAG, "Send states"); | ||||
|   for (auto *binarysensortype : this->binarysensortype_) { | ||||
|     if (force_update || binarysensortype->get_needs_to_send_update()) | ||||
|       binarysensortype->send_state_to_nextion(); | ||||
|   | ||||
| @@ -7,6 +7,17 @@ namespace number { | ||||
|  | ||||
| static const char *const TAG = "number"; | ||||
|  | ||||
| // Helper functions to reduce code size for logging | ||||
| void NumberCall::log_perform_warning_(const LogString *message) { | ||||
|   ESP_LOGW(TAG, "'%s': %s", this->parent_->get_name().c_str(), LOG_STR_ARG(message)); | ||||
| } | ||||
|  | ||||
| void NumberCall::log_perform_warning_value_range_(const LogString *comparison, const LogString *limit_type, float val, | ||||
|                                                   float limit) { | ||||
|   ESP_LOGW(TAG, "'%s': %f %s %s %f", this->parent_->get_name().c_str(), val, LOG_STR_ARG(comparison), | ||||
|            LOG_STR_ARG(limit_type), limit); | ||||
| } | ||||
|  | ||||
| NumberCall &NumberCall::set_value(float value) { return this->with_operation(NUMBER_OP_SET).with_value(value); } | ||||
|  | ||||
| NumberCall &NumberCall::number_increment(bool cycle) { | ||||
| @@ -42,7 +53,7 @@ void NumberCall::perform() { | ||||
|   const auto &traits = parent->traits; | ||||
|  | ||||
|   if (this->operation_ == NUMBER_OP_NONE) { | ||||
|     ESP_LOGW(TAG, "'%s' - NumberCall performed without selecting an operation", name); | ||||
|     this->log_perform_warning_(LOG_STR("No operation")); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
| @@ -51,28 +62,28 @@ void NumberCall::perform() { | ||||
|   float max_value = traits.get_max_value(); | ||||
|  | ||||
|   if (this->operation_ == NUMBER_OP_SET) { | ||||
|     ESP_LOGD(TAG, "'%s' - Setting number value", name); | ||||
|     ESP_LOGD(TAG, "'%s': Setting value", name); | ||||
|     if (!this->value_.has_value() || std::isnan(*this->value_)) { | ||||
|       ESP_LOGW(TAG, "'%s' - No value set for NumberCall", name); | ||||
|       this->log_perform_warning_(LOG_STR("No value")); | ||||
|       return; | ||||
|     } | ||||
|     target_value = this->value_.value(); | ||||
|   } else if (this->operation_ == NUMBER_OP_TO_MIN) { | ||||
|     if (std::isnan(min_value)) { | ||||
|       ESP_LOGW(TAG, "'%s' - Can't set to min value through NumberCall: no min_value defined", name); | ||||
|       this->log_perform_warning_(LOG_STR("min undefined")); | ||||
|     } else { | ||||
|       target_value = min_value; | ||||
|     } | ||||
|   } else if (this->operation_ == NUMBER_OP_TO_MAX) { | ||||
|     if (std::isnan(max_value)) { | ||||
|       ESP_LOGW(TAG, "'%s' - Can't set to max value through NumberCall: no max_value defined", name); | ||||
|       this->log_perform_warning_(LOG_STR("max undefined")); | ||||
|     } else { | ||||
|       target_value = max_value; | ||||
|     } | ||||
|   } else if (this->operation_ == NUMBER_OP_INCREMENT) { | ||||
|     ESP_LOGD(TAG, "'%s' - Increment number, with%s cycling", name, this->cycle_ ? "" : "out"); | ||||
|     ESP_LOGD(TAG, "'%s': Increment with%s cycling", name, this->cycle_ ? "" : "out"); | ||||
|     if (!parent->has_state()) { | ||||
|       ESP_LOGW(TAG, "'%s' - Can't increment number through NumberCall: no active state to modify", name); | ||||
|       this->log_perform_warning_(LOG_STR("Can't increment, no state")); | ||||
|       return; | ||||
|     } | ||||
|     auto step = traits.get_step(); | ||||
| @@ -85,9 +96,9 @@ void NumberCall::perform() { | ||||
|       } | ||||
|     } | ||||
|   } else if (this->operation_ == NUMBER_OP_DECREMENT) { | ||||
|     ESP_LOGD(TAG, "'%s' - Decrement number, with%s cycling", name, this->cycle_ ? "" : "out"); | ||||
|     ESP_LOGD(TAG, "'%s': Decrement with%s cycling", name, this->cycle_ ? "" : "out"); | ||||
|     if (!parent->has_state()) { | ||||
|       ESP_LOGW(TAG, "'%s' - Can't decrement number through NumberCall: no active state to modify", name); | ||||
|       this->log_perform_warning_(LOG_STR("Can't decrement, no state")); | ||||
|       return; | ||||
|     } | ||||
|     auto step = traits.get_step(); | ||||
| @@ -102,15 +113,15 @@ void NumberCall::perform() { | ||||
|   } | ||||
|  | ||||
|   if (target_value < min_value) { | ||||
|     ESP_LOGW(TAG, "'%s' - Value %f must not be less than minimum %f", name, target_value, min_value); | ||||
|     this->log_perform_warning_value_range_(LOG_STR("<"), LOG_STR("min"), target_value, min_value); | ||||
|     return; | ||||
|   } | ||||
|   if (target_value > max_value) { | ||||
|     ESP_LOGW(TAG, "'%s' - Value %f must not be greater than maximum %f", name, target_value, max_value); | ||||
|     this->log_perform_warning_value_range_(LOG_STR(">"), LOG_STR("max"), target_value, max_value); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   ESP_LOGD(TAG, "  New number value: %f", target_value); | ||||
|   ESP_LOGD(TAG, "  New value: %f", target_value); | ||||
|   this->parent_->control(target_value); | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "esphome/core/helpers.h" | ||||
| #include "esphome/core/log.h" | ||||
| #include "number_traits.h" | ||||
|  | ||||
| namespace esphome { | ||||
| @@ -33,6 +34,10 @@ class NumberCall { | ||||
|   NumberCall &with_cycle(bool cycle); | ||||
|  | ||||
|  protected: | ||||
|   void log_perform_warning_(const LogString *message); | ||||
|   void log_perform_warning_value_range_(const LogString *comparison, const LogString *limit_type, float val, | ||||
|                                         float limit); | ||||
|  | ||||
|   Number *const parent_; | ||||
|   NumberOperation operation_{NUMBER_OP_NONE}; | ||||
|   optional<float> value_; | ||||
|   | ||||
| @@ -117,7 +117,8 @@ int HOT BmpDecoder::decode(uint8_t *buffer, size_t size) { | ||||
|         this->paint_index_++; | ||||
|         this->current_index_ += 3; | ||||
|         index += 3; | ||||
|         if (x == this->width_ - 1 && this->padding_bytes_ > 0) { | ||||
|         size_t last_col = static_cast<size_t>(this->width_) - 1; | ||||
|         if (x == last_col && this->padding_bytes_ > 0) { | ||||
|           index += this->padding_bytes_; | ||||
|           this->current_index_ += this->padding_bytes_; | ||||
|         } | ||||
|   | ||||
| @@ -25,8 +25,10 @@ static int draw_callback(JPEGDRAW *jpeg) { | ||||
|   // to avoid crashing. | ||||
|   App.feed_wdt(); | ||||
|   size_t position = 0; | ||||
|   for (size_t y = 0; y < jpeg->iHeight; y++) { | ||||
|     for (size_t x = 0; x < jpeg->iWidth; x++) { | ||||
|   size_t height = static_cast<size_t>(jpeg->iHeight); | ||||
|   size_t width = static_cast<size_t>(jpeg->iWidth); | ||||
|   for (size_t y = 0; y < height; y++) { | ||||
|     for (size_t x = 0; x < width; x++) { | ||||
|       auto rg = decode_value(jpeg->pPixels[position++]); | ||||
|       auto ba = decode_value(jpeg->pPixels[position++]); | ||||
|       Color color(rg[1], rg[0], ba[1], ba[0]); | ||||
|   | ||||
| @@ -143,11 +143,10 @@ void OpenThreadSrpComponent::setup() { | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   // Copy the mdns services to our local instance so that the c_str pointers remain valid for the lifetime of this | ||||
|   // component | ||||
|   this->mdns_services_ = this->mdns_->get_services(); | ||||
|   ESP_LOGD(TAG, "Setting up SRP services. count = %d\n", this->mdns_services_.size()); | ||||
|   for (const auto &service : this->mdns_services_) { | ||||
|   // Get mdns services and copy their data (strings are copied with strdup below) | ||||
|   const auto &mdns_services = this->mdns_->get_services(); | ||||
|   ESP_LOGD(TAG, "Setting up SRP services. count = %d\n", mdns_services.size()); | ||||
|   for (const auto &service : mdns_services) { | ||||
|     otSrpClientBuffersServiceEntry *entry = otSrpClientBuffersAllocateService(instance); | ||||
|     if (!entry) { | ||||
|       ESP_LOGW(TAG, "Failed to allocate service entry"); | ||||
|   | ||||
| @@ -57,7 +57,6 @@ class OpenThreadSrpComponent : public Component { | ||||
|  | ||||
|  protected: | ||||
|   esphome::mdns::MDNSComponent *mdns_{nullptr}; | ||||
|   std::vector<esphome::mdns::MDNSService> mdns_services_; | ||||
|   std::vector<std::unique_ptr<uint8_t[]>> memory_pool_; | ||||
|   void *pool_alloc_(size_t size); | ||||
| }; | ||||
|   | ||||
| @@ -104,7 +104,7 @@ float PIDController::weighted_average_(std::deque<float> &list, float new_value, | ||||
|   list.push_front(new_value); | ||||
|  | ||||
|   // keep only 'samples' readings, by popping off the back of the list | ||||
|   while (list.size() > samples) | ||||
|   while (samples > 0 && list.size() > static_cast<size_t>(samples)) | ||||
|     list.pop_back(); | ||||
|  | ||||
|   // calculate and return the average of all values in the list | ||||
|   | ||||
| @@ -110,21 +110,21 @@ std::string PrometheusHandler::relabel_name_(EntityBase *obj) { | ||||
|  | ||||
| void PrometheusHandler::add_area_label_(AsyncResponseStream *stream, std::string &area) { | ||||
|   if (!area.empty()) { | ||||
|     stream->print(F("\",area=\"")); | ||||
|     stream->print(ESPHOME_F("\",area=\"")); | ||||
|     stream->print(area.c_str()); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void PrometheusHandler::add_node_label_(AsyncResponseStream *stream, std::string &node) { | ||||
|   if (!node.empty()) { | ||||
|     stream->print(F("\",node=\"")); | ||||
|     stream->print(ESPHOME_F("\",node=\"")); | ||||
|     stream->print(node.c_str()); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void PrometheusHandler::add_friendly_name_label_(AsyncResponseStream *stream, std::string &friendly_name) { | ||||
|   if (!friendly_name.empty()) { | ||||
|     stream->print(F("\",friendly_name=\"")); | ||||
|     stream->print(ESPHOME_F("\",friendly_name=\"")); | ||||
|     stream->print(friendly_name.c_str()); | ||||
|   } | ||||
| } | ||||
| @@ -132,8 +132,8 @@ void PrometheusHandler::add_friendly_name_label_(AsyncResponseStream *stream, st | ||||
| // Type-specific implementation | ||||
| #ifdef USE_SENSOR | ||||
| void PrometheusHandler::sensor_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_sensor_value gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_sensor_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_sensor_value gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_sensor_failed gauge\n")); | ||||
| } | ||||
| void PrometheusHandler::sensor_row_(AsyncResponseStream *stream, sensor::Sensor *obj, std::string &area, | ||||
|                                     std::string &node, std::string &friendly_name) { | ||||
| @@ -141,37 +141,37 @@ void PrometheusHandler::sensor_row_(AsyncResponseStream *stream, sensor::Sensor | ||||
|     return; | ||||
|   if (!std::isnan(obj->state)) { | ||||
|     // We have a valid value, output this value | ||||
|     stream->print(F("esphome_sensor_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_sensor_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 0\n")); | ||||
|     stream->print(ESPHOME_F("\"} 0\n")); | ||||
|     // Data itself | ||||
|     stream->print(F("esphome_sensor_value{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_sensor_value{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\",unit=\"")); | ||||
|     stream->print(ESPHOME_F("\",unit=\"")); | ||||
|     stream->print(obj->get_unit_of_measurement().c_str()); | ||||
|     stream->print(F("\"} ")); | ||||
|     stream->print(ESPHOME_F("\"} ")); | ||||
|     stream->print(value_accuracy_to_string(obj->state, obj->get_accuracy_decimals()).c_str()); | ||||
|     stream->print(F("\n")); | ||||
|     stream->print(ESPHOME_F("\n")); | ||||
|   } else { | ||||
|     // Invalid state | ||||
|     stream->print(F("esphome_sensor_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_sensor_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 1\n")); | ||||
|     stream->print(ESPHOME_F("\"} 1\n")); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
| @@ -179,8 +179,8 @@ void PrometheusHandler::sensor_row_(AsyncResponseStream *stream, sensor::Sensor | ||||
| // Type-specific implementation | ||||
| #ifdef USE_BINARY_SENSOR | ||||
| void PrometheusHandler::binary_sensor_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_binary_sensor_value gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_binary_sensor_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_binary_sensor_value gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_binary_sensor_failed gauge\n")); | ||||
| } | ||||
| void PrometheusHandler::binary_sensor_row_(AsyncResponseStream *stream, binary_sensor::BinarySensor *obj, | ||||
|                                            std::string &area, std::string &node, std::string &friendly_name) { | ||||
| @@ -188,204 +188,204 @@ void PrometheusHandler::binary_sensor_row_(AsyncResponseStream *stream, binary_s | ||||
|     return; | ||||
|   if (obj->has_state()) { | ||||
|     // We have a valid value, output this value | ||||
|     stream->print(F("esphome_binary_sensor_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_binary_sensor_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 0\n")); | ||||
|     stream->print(ESPHOME_F("\"} 0\n")); | ||||
|     // Data itself | ||||
|     stream->print(F("esphome_binary_sensor_value{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_binary_sensor_value{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} ")); | ||||
|     stream->print(ESPHOME_F("\"} ")); | ||||
|     stream->print(obj->state); | ||||
|     stream->print(F("\n")); | ||||
|     stream->print(ESPHOME_F("\n")); | ||||
|   } else { | ||||
|     // Invalid state | ||||
|     stream->print(F("esphome_binary_sensor_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_binary_sensor_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 1\n")); | ||||
|     stream->print(ESPHOME_F("\"} 1\n")); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_FAN | ||||
| void PrometheusHandler::fan_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_fan_value gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_fan_failed gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_fan_speed gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_fan_oscillation gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_fan_value gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_fan_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_fan_speed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_fan_oscillation gauge\n")); | ||||
| } | ||||
| void PrometheusHandler::fan_row_(AsyncResponseStream *stream, fan::Fan *obj, std::string &area, std::string &node, | ||||
|                                  std::string &friendly_name) { | ||||
|   if (obj->is_internal() && !this->include_internal_) | ||||
|     return; | ||||
|   stream->print(F("esphome_fan_failed{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_fan_failed{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\"} 0\n")); | ||||
|   stream->print(ESPHOME_F("\"} 0\n")); | ||||
|   // Data itself | ||||
|   stream->print(F("esphome_fan_value{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_fan_value{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\"} ")); | ||||
|   stream->print(ESPHOME_F("\"} ")); | ||||
|   stream->print(obj->state); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
|   // Speed if available | ||||
|   if (obj->get_traits().supports_speed()) { | ||||
|     stream->print(F("esphome_fan_speed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_fan_speed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} ")); | ||||
|     stream->print(ESPHOME_F("\"} ")); | ||||
|     stream->print(obj->speed); | ||||
|     stream->print(F("\n")); | ||||
|     stream->print(ESPHOME_F("\n")); | ||||
|   } | ||||
|   // Oscillation if available | ||||
|   if (obj->get_traits().supports_oscillation()) { | ||||
|     stream->print(F("esphome_fan_oscillation{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_fan_oscillation{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} ")); | ||||
|     stream->print(ESPHOME_F("\"} ")); | ||||
|     stream->print(obj->oscillating); | ||||
|     stream->print(F("\n")); | ||||
|     stream->print(ESPHOME_F("\n")); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_LIGHT | ||||
| void PrometheusHandler::light_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_light_state gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_light_color gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_light_effect_active gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_light_state gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_light_color gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_light_effect_active gauge\n")); | ||||
| } | ||||
| void PrometheusHandler::light_row_(AsyncResponseStream *stream, light::LightState *obj, std::string &area, | ||||
|                                    std::string &node, std::string &friendly_name) { | ||||
|   if (obj->is_internal() && !this->include_internal_) | ||||
|     return; | ||||
|   // State | ||||
|   stream->print(F("esphome_light_state{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_light_state{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\"} ")); | ||||
|   stream->print(ESPHOME_F("\"} ")); | ||||
|   stream->print(obj->remote_values.is_on()); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
|   // Brightness and RGBW | ||||
|   light::LightColorValues color = obj->current_values; | ||||
|   float brightness, r, g, b, w; | ||||
|   color.as_brightness(&brightness); | ||||
|   color.as_rgbw(&r, &g, &b, &w); | ||||
|   stream->print(F("esphome_light_color{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_light_color{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\",channel=\"brightness\"} ")); | ||||
|   stream->print(ESPHOME_F("\",channel=\"brightness\"} ")); | ||||
|   stream->print(brightness); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(F("esphome_light_color{id=\"")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
|   stream->print(ESPHOME_F("esphome_light_color{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\",channel=\"r\"} ")); | ||||
|   stream->print(ESPHOME_F("\",channel=\"r\"} ")); | ||||
|   stream->print(r); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(F("esphome_light_color{id=\"")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
|   stream->print(ESPHOME_F("esphome_light_color{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\",channel=\"g\"} ")); | ||||
|   stream->print(ESPHOME_F("\",channel=\"g\"} ")); | ||||
|   stream->print(g); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(F("esphome_light_color{id=\"")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
|   stream->print(ESPHOME_F("esphome_light_color{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\",channel=\"b\"} ")); | ||||
|   stream->print(ESPHOME_F("\",channel=\"b\"} ")); | ||||
|   stream->print(b); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(F("esphome_light_color{id=\"")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
|   stream->print(ESPHOME_F("esphome_light_color{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\",channel=\"w\"} ")); | ||||
|   stream->print(ESPHOME_F("\",channel=\"w\"} ")); | ||||
|   stream->print(w); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
|   // Effect | ||||
|   std::string effect = obj->get_effect_name(); | ||||
|   if (effect == "None") { | ||||
|     stream->print(F("esphome_light_effect_active{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_light_effect_active{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\",effect=\"None\"} 0\n")); | ||||
|     stream->print(ESPHOME_F("\",effect=\"None\"} 0\n")); | ||||
|   } else { | ||||
|     stream->print(F("esphome_light_effect_active{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_light_effect_active{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\",effect=\"")); | ||||
|     stream->print(ESPHOME_F("\",effect=\"")); | ||||
|     stream->print(effect.c_str()); | ||||
|     stream->print(F("\"} 1\n")); | ||||
|     stream->print(ESPHOME_F("\"} 1\n")); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_COVER | ||||
| void PrometheusHandler::cover_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_cover_value gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_cover_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_cover_value gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_cover_failed gauge\n")); | ||||
| } | ||||
| void PrometheusHandler::cover_row_(AsyncResponseStream *stream, cover::Cover *obj, std::string &area, std::string &node, | ||||
|                                    std::string &friendly_name) { | ||||
| @@ -393,118 +393,118 @@ void PrometheusHandler::cover_row_(AsyncResponseStream *stream, cover::Cover *ob | ||||
|     return; | ||||
|   if (!std::isnan(obj->position)) { | ||||
|     // We have a valid value, output this value | ||||
|     stream->print(F("esphome_cover_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_cover_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 0\n")); | ||||
|     stream->print(ESPHOME_F("\"} 0\n")); | ||||
|     // Data itself | ||||
|     stream->print(F("esphome_cover_value{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_cover_value{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} ")); | ||||
|     stream->print(ESPHOME_F("\"} ")); | ||||
|     stream->print(obj->position); | ||||
|     stream->print(F("\n")); | ||||
|     stream->print(ESPHOME_F("\n")); | ||||
|     if (obj->get_traits().get_supports_tilt()) { | ||||
|       stream->print(F("esphome_cover_tilt{id=\"")); | ||||
|       stream->print(ESPHOME_F("esphome_cover_tilt{id=\"")); | ||||
|       stream->print(relabel_id_(obj).c_str()); | ||||
|       add_area_label_(stream, area); | ||||
|       add_node_label_(stream, node); | ||||
|       add_friendly_name_label_(stream, friendly_name); | ||||
|       stream->print(F("\",name=\"")); | ||||
|       stream->print(ESPHOME_F("\",name=\"")); | ||||
|       stream->print(relabel_name_(obj).c_str()); | ||||
|       stream->print(F("\"} ")); | ||||
|       stream->print(ESPHOME_F("\"} ")); | ||||
|       stream->print(obj->tilt); | ||||
|       stream->print(F("\n")); | ||||
|       stream->print(ESPHOME_F("\n")); | ||||
|     } | ||||
|   } else { | ||||
|     // Invalid state | ||||
|     stream->print(F("esphome_cover_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_cover_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 1\n")); | ||||
|     stream->print(ESPHOME_F("\"} 1\n")); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_SWITCH | ||||
| void PrometheusHandler::switch_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_switch_value gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_switch_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_switch_value gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_switch_failed gauge\n")); | ||||
| } | ||||
| void PrometheusHandler::switch_row_(AsyncResponseStream *stream, switch_::Switch *obj, std::string &area, | ||||
|                                     std::string &node, std::string &friendly_name) { | ||||
|   if (obj->is_internal() && !this->include_internal_) | ||||
|     return; | ||||
|   stream->print(F("esphome_switch_failed{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_switch_failed{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\"} 0\n")); | ||||
|   stream->print(ESPHOME_F("\"} 0\n")); | ||||
|   // Data itself | ||||
|   stream->print(F("esphome_switch_value{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_switch_value{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\"} ")); | ||||
|   stream->print(ESPHOME_F("\"} ")); | ||||
|   stream->print(obj->state); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_LOCK | ||||
| void PrometheusHandler::lock_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_lock_value gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_lock_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_lock_value gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_lock_failed gauge\n")); | ||||
| } | ||||
| void PrometheusHandler::lock_row_(AsyncResponseStream *stream, lock::Lock *obj, std::string &area, std::string &node, | ||||
|                                   std::string &friendly_name) { | ||||
|   if (obj->is_internal() && !this->include_internal_) | ||||
|     return; | ||||
|   stream->print(F("esphome_lock_failed{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_lock_failed{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\"} 0\n")); | ||||
|   stream->print(ESPHOME_F("\"} 0\n")); | ||||
|   // Data itself | ||||
|   stream->print(F("esphome_lock_value{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_lock_value{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\"} ")); | ||||
|   stream->print(ESPHOME_F("\"} ")); | ||||
|   stream->print(obj->state); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
| } | ||||
| #endif | ||||
|  | ||||
| // Type-specific implementation | ||||
| #ifdef USE_TEXT_SENSOR | ||||
| void PrometheusHandler::text_sensor_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_text_sensor_value gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_text_sensor_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_text_sensor_value gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_text_sensor_failed gauge\n")); | ||||
| } | ||||
| void PrometheusHandler::text_sensor_row_(AsyncResponseStream *stream, text_sensor::TextSensor *obj, std::string &area, | ||||
|                                          std::string &node, std::string &friendly_name) { | ||||
| @@ -512,37 +512,37 @@ void PrometheusHandler::text_sensor_row_(AsyncResponseStream *stream, text_senso | ||||
|     return; | ||||
|   if (obj->has_state()) { | ||||
|     // We have a valid value, output this value | ||||
|     stream->print(F("esphome_text_sensor_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_text_sensor_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 0\n")); | ||||
|     stream->print(ESPHOME_F("\"} 0\n")); | ||||
|     // Data itself | ||||
|     stream->print(F("esphome_text_sensor_value{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_text_sensor_value{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\",value=\"")); | ||||
|     stream->print(ESPHOME_F("\",value=\"")); | ||||
|     stream->print(obj->state.c_str()); | ||||
|     stream->print(F("\"} ")); | ||||
|     stream->print(F("1.0")); | ||||
|     stream->print(F("\n")); | ||||
|     stream->print(ESPHOME_F("\"} ")); | ||||
|     stream->print(ESPHOME_F("1.0")); | ||||
|     stream->print(ESPHOME_F("\n")); | ||||
|   } else { | ||||
|     // Invalid state | ||||
|     stream->print(F("esphome_text_sensor_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_text_sensor_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 1\n")); | ||||
|     stream->print(ESPHOME_F("\"} 1\n")); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
| @@ -550,8 +550,8 @@ void PrometheusHandler::text_sensor_row_(AsyncResponseStream *stream, text_senso | ||||
| // Type-specific implementation | ||||
| #ifdef USE_NUMBER | ||||
| void PrometheusHandler::number_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_number_value gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_number_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_number_value gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_number_failed gauge\n")); | ||||
| } | ||||
| void PrometheusHandler::number_row_(AsyncResponseStream *stream, number::Number *obj, std::string &area, | ||||
|                                     std::string &node, std::string &friendly_name) { | ||||
| @@ -559,43 +559,43 @@ void PrometheusHandler::number_row_(AsyncResponseStream *stream, number::Number | ||||
|     return; | ||||
|   if (!std::isnan(obj->state)) { | ||||
|     // We have a valid value, output this value | ||||
|     stream->print(F("esphome_number_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_number_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 0\n")); | ||||
|     stream->print(ESPHOME_F("\"} 0\n")); | ||||
|     // Data itself | ||||
|     stream->print(F("esphome_number_value{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_number_value{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} ")); | ||||
|     stream->print(ESPHOME_F("\"} ")); | ||||
|     stream->print(obj->state); | ||||
|     stream->print(F("\n")); | ||||
|     stream->print(ESPHOME_F("\n")); | ||||
|   } else { | ||||
|     // Invalid state | ||||
|     stream->print(F("esphome_number_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_number_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 1\n")); | ||||
|     stream->print(ESPHOME_F("\"} 1\n")); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_SELECT | ||||
| void PrometheusHandler::select_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_select_value gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_select_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_select_value gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_select_failed gauge\n")); | ||||
| } | ||||
| void PrometheusHandler::select_row_(AsyncResponseStream *stream, select::Select *obj, std::string &area, | ||||
|                                     std::string &node, std::string &friendly_name) { | ||||
| @@ -603,105 +603,105 @@ void PrometheusHandler::select_row_(AsyncResponseStream *stream, select::Select | ||||
|     return; | ||||
|   if (obj->has_state()) { | ||||
|     // We have a valid value, output this value | ||||
|     stream->print(F("esphome_select_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_select_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 0\n")); | ||||
|     stream->print(ESPHOME_F("\"} 0\n")); | ||||
|     // Data itself | ||||
|     stream->print(F("esphome_select_value{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_select_value{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\",value=\"")); | ||||
|     stream->print(ESPHOME_F("\",value=\"")); | ||||
|     stream->print(obj->state.c_str()); | ||||
|     stream->print(F("\"} ")); | ||||
|     stream->print(F("1.0")); | ||||
|     stream->print(F("\n")); | ||||
|     stream->print(ESPHOME_F("\"} ")); | ||||
|     stream->print(ESPHOME_F("1.0")); | ||||
|     stream->print(ESPHOME_F("\n")); | ||||
|   } else { | ||||
|     // Invalid state | ||||
|     stream->print(F("esphome_select_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_select_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 1\n")); | ||||
|     stream->print(ESPHOME_F("\"} 1\n")); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_MEDIA_PLAYER | ||||
| void PrometheusHandler::media_player_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_media_player_state_value gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_media_player_volume gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_media_player_is_muted gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_media_player_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_media_player_state_value gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_media_player_volume gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_media_player_is_muted gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_media_player_failed gauge\n")); | ||||
| } | ||||
| void PrometheusHandler::media_player_row_(AsyncResponseStream *stream, media_player::MediaPlayer *obj, | ||||
|                                           std::string &area, std::string &node, std::string &friendly_name) { | ||||
|   if (obj->is_internal() && !this->include_internal_) | ||||
|     return; | ||||
|   stream->print(F("esphome_media_player_failed{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_media_player_failed{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\"} 0\n")); | ||||
|   stream->print(ESPHOME_F("\"} 0\n")); | ||||
|   // Data itself | ||||
|   stream->print(F("esphome_media_player_state_value{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_media_player_state_value{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\",value=\"")); | ||||
|   stream->print(ESPHOME_F("\",value=\"")); | ||||
|   stream->print(media_player::media_player_state_to_string(obj->state)); | ||||
|   stream->print(F("\"} ")); | ||||
|   stream->print(F("1.0")); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(F("esphome_media_player_volume{id=\"")); | ||||
|   stream->print(ESPHOME_F("\"} ")); | ||||
|   stream->print(ESPHOME_F("1.0")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
|   stream->print(ESPHOME_F("esphome_media_player_volume{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\"} ")); | ||||
|   stream->print(ESPHOME_F("\"} ")); | ||||
|   stream->print(obj->volume); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(F("esphome_media_player_is_muted{id=\"")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
|   stream->print(ESPHOME_F("esphome_media_player_is_muted{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\"} ")); | ||||
|   stream->print(ESPHOME_F("\"} ")); | ||||
|   if (obj->is_muted()) { | ||||
|     stream->print(F("1.0")); | ||||
|     stream->print(ESPHOME_F("1.0")); | ||||
|   } else { | ||||
|     stream->print(F("0.0")); | ||||
|     stream->print(ESPHOME_F("0.0")); | ||||
|   } | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_UPDATE | ||||
| void PrometheusHandler::update_entity_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_update_entity_state gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_update_entity_info gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_update_entity_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_update_entity_state gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_update_entity_info gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_update_entity_failed gauge\n")); | ||||
| } | ||||
|  | ||||
| void PrometheusHandler::handle_update_state_(AsyncResponseStream *stream, update::UpdateState state) { | ||||
| @@ -730,168 +730,168 @@ void PrometheusHandler::update_entity_row_(AsyncResponseStream *stream, update:: | ||||
|     return; | ||||
|   if (obj->has_state()) { | ||||
|     // We have a valid value, output this value | ||||
|     stream->print(F("esphome_update_entity_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_update_entity_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 0\n")); | ||||
|     stream->print(ESPHOME_F("\"} 0\n")); | ||||
|     // First update state | ||||
|     stream->print(F("esphome_update_entity_state{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_update_entity_state{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\",value=\"")); | ||||
|     stream->print(ESPHOME_F("\",value=\"")); | ||||
|     handle_update_state_(stream, obj->state); | ||||
|     stream->print(F("\"} ")); | ||||
|     stream->print(F("1.0")); | ||||
|     stream->print(F("\n")); | ||||
|     stream->print(ESPHOME_F("\"} ")); | ||||
|     stream->print(ESPHOME_F("1.0")); | ||||
|     stream->print(ESPHOME_F("\n")); | ||||
|     // Next update info | ||||
|     stream->print(F("esphome_update_entity_info{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_update_entity_info{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\",current_version=\"")); | ||||
|     stream->print(ESPHOME_F("\",current_version=\"")); | ||||
|     stream->print(obj->update_info.current_version.c_str()); | ||||
|     stream->print(F("\",latest_version=\"")); | ||||
|     stream->print(ESPHOME_F("\",latest_version=\"")); | ||||
|     stream->print(obj->update_info.latest_version.c_str()); | ||||
|     stream->print(F("\",title=\"")); | ||||
|     stream->print(ESPHOME_F("\",title=\"")); | ||||
|     stream->print(obj->update_info.title.c_str()); | ||||
|     stream->print(F("\"} ")); | ||||
|     stream->print(F("1.0")); | ||||
|     stream->print(F("\n")); | ||||
|     stream->print(ESPHOME_F("\"} ")); | ||||
|     stream->print(ESPHOME_F("1.0")); | ||||
|     stream->print(ESPHOME_F("\n")); | ||||
|   } else { | ||||
|     // Invalid state | ||||
|     stream->print(F("esphome_update_entity_failed{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_update_entity_failed{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} 1\n")); | ||||
|     stream->print(ESPHOME_F("\"} 1\n")); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_VALVE | ||||
| void PrometheusHandler::valve_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_valve_operation gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_valve_failed gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_valve_position gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_valve_operation gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_valve_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_valve_position gauge\n")); | ||||
| } | ||||
|  | ||||
| void PrometheusHandler::valve_row_(AsyncResponseStream *stream, valve::Valve *obj, std::string &area, std::string &node, | ||||
|                                    std::string &friendly_name) { | ||||
|   if (obj->is_internal() && !this->include_internal_) | ||||
|     return; | ||||
|   stream->print(F("esphome_valve_failed{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_valve_failed{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\"} 0\n")); | ||||
|   stream->print(ESPHOME_F("\"} 0\n")); | ||||
|   // Data itself | ||||
|   stream->print(F("esphome_valve_operation{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_valve_operation{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\",operation=\"")); | ||||
|   stream->print(ESPHOME_F("\",operation=\"")); | ||||
|   stream->print(valve::valve_operation_to_str(obj->current_operation)); | ||||
|   stream->print(F("\"} ")); | ||||
|   stream->print(F("1.0")); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(ESPHOME_F("\"} ")); | ||||
|   stream->print(ESPHOME_F("1.0")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
|   // Now see if position is supported | ||||
|   if (obj->get_traits().get_supports_position()) { | ||||
|     stream->print(F("esphome_valve_position{id=\"")); | ||||
|     stream->print(ESPHOME_F("esphome_valve_position{id=\"")); | ||||
|     stream->print(relabel_id_(obj).c_str()); | ||||
|     add_area_label_(stream, area); | ||||
|     add_node_label_(stream, node); | ||||
|     add_friendly_name_label_(stream, friendly_name); | ||||
|     stream->print(F("\",name=\"")); | ||||
|     stream->print(ESPHOME_F("\",name=\"")); | ||||
|     stream->print(relabel_name_(obj).c_str()); | ||||
|     stream->print(F("\"} ")); | ||||
|     stream->print(ESPHOME_F("\"} ")); | ||||
|     stream->print(obj->position); | ||||
|     stream->print(F("\n")); | ||||
|     stream->print(ESPHOME_F("\n")); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #ifdef USE_CLIMATE | ||||
| void PrometheusHandler::climate_type_(AsyncResponseStream *stream) { | ||||
|   stream->print(F("#TYPE esphome_climate_setting gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_climate_value gauge\n")); | ||||
|   stream->print(F("#TYPE esphome_climate_failed gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_climate_setting gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_climate_value gauge\n")); | ||||
|   stream->print(ESPHOME_F("#TYPE esphome_climate_failed gauge\n")); | ||||
| } | ||||
|  | ||||
| void PrometheusHandler::climate_setting_row_(AsyncResponseStream *stream, climate::Climate *obj, std::string &area, | ||||
|                                              std::string &node, std::string &friendly_name, std::string &setting, | ||||
|                                              const LogString *setting_value) { | ||||
|   stream->print(F("esphome_climate_setting{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_climate_setting{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\",category=\"")); | ||||
|   stream->print(ESPHOME_F("\",category=\"")); | ||||
|   stream->print(setting.c_str()); | ||||
|   stream->print(F("\",setting_value=\"")); | ||||
|   stream->print(ESPHOME_F("\",setting_value=\"")); | ||||
|   stream->print(LOG_STR_ARG(setting_value)); | ||||
|   stream->print(F("\"} ")); | ||||
|   stream->print(F("1.0")); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(ESPHOME_F("\"} ")); | ||||
|   stream->print(ESPHOME_F("1.0")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
| } | ||||
|  | ||||
| void PrometheusHandler::climate_value_row_(AsyncResponseStream *stream, climate::Climate *obj, std::string &area, | ||||
|                                            std::string &node, std::string &friendly_name, std::string &category, | ||||
|                                            std::string &climate_value) { | ||||
|   stream->print(F("esphome_climate_value{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_climate_value{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\",category=\"")); | ||||
|   stream->print(ESPHOME_F("\",category=\"")); | ||||
|   stream->print(category.c_str()); | ||||
|   stream->print(F("\"} ")); | ||||
|   stream->print(ESPHOME_F("\"} ")); | ||||
|   stream->print(climate_value.c_str()); | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
| } | ||||
|  | ||||
| void PrometheusHandler::climate_failed_row_(AsyncResponseStream *stream, climate::Climate *obj, std::string &area, | ||||
|                                             std::string &node, std::string &friendly_name, std::string &category, | ||||
|                                             bool is_failed_value) { | ||||
|   stream->print(F("esphome_climate_failed{id=\"")); | ||||
|   stream->print(ESPHOME_F("esphome_climate_failed{id=\"")); | ||||
|   stream->print(relabel_id_(obj).c_str()); | ||||
|   add_area_label_(stream, area); | ||||
|   add_node_label_(stream, node); | ||||
|   add_friendly_name_label_(stream, friendly_name); | ||||
|   stream->print(F("\",name=\"")); | ||||
|   stream->print(ESPHOME_F("\",name=\"")); | ||||
|   stream->print(relabel_name_(obj).c_str()); | ||||
|   stream->print(F("\",category=\"")); | ||||
|   stream->print(ESPHOME_F("\",category=\"")); | ||||
|   stream->print(category.c_str()); | ||||
|   stream->print(F("\"} ")); | ||||
|   stream->print(ESPHOME_F("\"} ")); | ||||
|   if (is_failed_value) { | ||||
|     stream->print(F("1.0")); | ||||
|     stream->print(ESPHOME_F("1.0")); | ||||
|   } else { | ||||
|     stream->print(F("0.0")); | ||||
|     stream->print(ESPHOME_F("0.0")); | ||||
|   } | ||||
|   stream->print(F("\n")); | ||||
|   stream->print(ESPHOME_F("\n")); | ||||
| } | ||||
|  | ||||
| void PrometheusHandler::climate_row_(AsyncResponseStream *stream, climate::Climate *obj, std::string &area, | ||||
|   | ||||
| @@ -62,6 +62,11 @@ SPIRAM_SPEEDS = { | ||||
| } | ||||
|  | ||||
|  | ||||
| def supported() -> bool: | ||||
|     variant = get_esp32_variant() | ||||
|     return variant in SPIRAM_MODES | ||||
|  | ||||
|  | ||||
| def validate_psram_mode(config): | ||||
|     esp32_config = fv.full_config.get()[PLATFORM_ESP32] | ||||
|     if config[CONF_SPEED] == "120MHZ": | ||||
| @@ -95,7 +100,7 @@ def get_config_schema(config): | ||||
|     variant = get_esp32_variant() | ||||
|     speeds = [f"{s}MHZ" for s in SPIRAM_SPEEDS.get(variant, [])] | ||||
|     if not speeds: | ||||
|         return cv.Invalid("PSRAM is not supported on this chip") | ||||
|         raise cv.Invalid("PSRAM is not supported on this chip") | ||||
|     modes = SPIRAM_MODES[variant] | ||||
|     return cv.Schema( | ||||
|         { | ||||
|   | ||||
| @@ -8,6 +8,7 @@ namespace esphome { | ||||
| namespace qmc5883l { | ||||
|  | ||||
| static const char *const TAG = "qmc5883l"; | ||||
|  | ||||
| static const uint8_t QMC5883L_ADDRESS = 0x0D; | ||||
|  | ||||
| static const uint8_t QMC5883L_REGISTER_DATA_X_LSB = 0x00; | ||||
| @@ -32,6 +33,10 @@ void QMC5883LComponent::setup() { | ||||
|   } | ||||
|   delay(10); | ||||
|  | ||||
|   if (this->drdy_pin_) { | ||||
|     this->drdy_pin_->setup(); | ||||
|   } | ||||
|  | ||||
|   uint8_t control_1 = 0; | ||||
|   control_1 |= 0b01 << 0;  // MODE (Mode) -> 0b00=standby, 0b01=continuous | ||||
|   control_1 |= this->datarate_ << 2; | ||||
| @@ -64,6 +69,7 @@ void QMC5883LComponent::setup() { | ||||
|     high_freq_.start(); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void QMC5883LComponent::dump_config() { | ||||
|   ESP_LOGCONFIG(TAG, "QMC5883L:"); | ||||
|   LOG_I2C_DEVICE(this); | ||||
| @@ -77,11 +83,20 @@ void QMC5883LComponent::dump_config() { | ||||
|   LOG_SENSOR("  ", "Z Axis", this->z_sensor_); | ||||
|   LOG_SENSOR("  ", "Heading", this->heading_sensor_); | ||||
|   LOG_SENSOR("  ", "Temperature", this->temperature_sensor_); | ||||
|   LOG_PIN("  DRDY Pin: ", this->drdy_pin_); | ||||
| } | ||||
|  | ||||
| float QMC5883LComponent::get_setup_priority() const { return setup_priority::DATA; } | ||||
|  | ||||
| void QMC5883LComponent::update() { | ||||
|   i2c::ErrorCode err; | ||||
|   uint8_t status = false; | ||||
|  | ||||
|   // If DRDY pin is configured and the data is not ready return. | ||||
|   if (this->drdy_pin_ && !this->drdy_pin_->digital_read()) { | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   // Status byte gets cleared when data is read, so we have to read this first. | ||||
|   // If status and two axes are desired, it's possible to save one byte of traffic by enabling | ||||
|   // ROL_PNT in setup and reading 7 bytes starting at the status register. | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user