mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-31 23:21:54 +00:00 
			
		
		
		
	Merge branch 'integration' into memory_api
This commit is contained in:
		| @@ -27,7 +27,7 @@ void SNTPComponent::setup() { | |||||||
|   esp_sntp_setoperatingmode(ESP_SNTP_OPMODE_POLL); |   esp_sntp_setoperatingmode(ESP_SNTP_OPMODE_POLL); | ||||||
|   size_t i = 0; |   size_t i = 0; | ||||||
|   for (auto &server : this->servers_) { |   for (auto &server : this->servers_) { | ||||||
|     esp_sntp_setservername(i++, server.c_str()); |     esp_sntp_setservername(i++, server); | ||||||
|   } |   } | ||||||
|   esp_sntp_set_sync_interval(this->get_update_interval()); |   esp_sntp_set_sync_interval(this->get_update_interval()); | ||||||
|   esp_sntp_set_time_sync_notification_cb([](struct timeval *tv) { |   esp_sntp_set_time_sync_notification_cb([](struct timeval *tv) { | ||||||
| @@ -42,7 +42,16 @@ void SNTPComponent::setup() { | |||||||
|  |  | ||||||
|   size_t i = 0; |   size_t i = 0; | ||||||
|   for (auto &server : this->servers_) { |   for (auto &server : this->servers_) { | ||||||
|     sntp_setservername(i++, server.c_str()); | #if defined(USE_ESP8266) | ||||||
|  |     // On ESP8266, server is PGM_P pointing to PROGMEM | ||||||
|  |     // LWIP's sntp_setservername is not PROGMEM-aware, so copy to stack buffer first | ||||||
|  |     char server_buf[64]; | ||||||
|  |     strncpy_P(server_buf, server, sizeof(server_buf) - 1); | ||||||
|  |     server_buf[sizeof(server_buf) - 1] = '\0'; | ||||||
|  |     sntp_setservername(i++, server_buf); | ||||||
|  | #else | ||||||
|  |     sntp_setservername(i++, server); | ||||||
|  | #endif | ||||||
|   } |   } | ||||||
|  |  | ||||||
| #if defined(USE_ESP8266) | #if defined(USE_ESP8266) | ||||||
| @@ -59,7 +68,8 @@ void SNTPComponent::dump_config() { | |||||||
|   ESP_LOGCONFIG(TAG, "SNTP Time:"); |   ESP_LOGCONFIG(TAG, "SNTP Time:"); | ||||||
|   size_t i = 0; |   size_t i = 0; | ||||||
|   for (auto &server : this->servers_) { |   for (auto &server : this->servers_) { | ||||||
|     ESP_LOGCONFIG(TAG, "  Server %zu: '%s'", i++, server.c_str()); |     // LOG_STR_ARG handles both PROGMEM (ESP8266) and regular pointers | ||||||
|  |     ESP_LOGCONFIG(TAG, "  Server %zu: '%s'", i++, LOG_STR_ARG(server)); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| void SNTPComponent::update() { | void SNTPComponent::update() { | ||||||
|   | |||||||
| @@ -2,10 +2,18 @@ | |||||||
|  |  | ||||||
| #include "esphome/core/component.h" | #include "esphome/core/component.h" | ||||||
| #include "esphome/components/time/real_time_clock.h" | #include "esphome/components/time/real_time_clock.h" | ||||||
|  | #include <array> | ||||||
|  |  | ||||||
|  | #ifdef USE_ESP8266 | ||||||
|  | #include <pgmspace.h> | ||||||
|  | #endif | ||||||
|  |  | ||||||
| namespace esphome { | namespace esphome { | ||||||
| namespace sntp { | namespace sntp { | ||||||
|  |  | ||||||
|  | // Server count is calculated at compile time by Python codegen | ||||||
|  | // SNTP_SERVER_COUNT will always be defined | ||||||
|  |  | ||||||
| /// The SNTP component allows you to configure local timekeeping via Simple Network Time Protocol. | /// The SNTP component allows you to configure local timekeeping via Simple Network Time Protocol. | ||||||
| /// | /// | ||||||
| /// \note | /// \note | ||||||
| @@ -14,10 +22,7 @@ namespace sntp { | |||||||
| /// \see https://www.gnu.org/software/libc/manual/html_node/TZ-Variable.html | /// \see https://www.gnu.org/software/libc/manual/html_node/TZ-Variable.html | ||||||
| class SNTPComponent : public time::RealTimeClock { | class SNTPComponent : public time::RealTimeClock { | ||||||
|  public: |  public: | ||||||
|   SNTPComponent(const std::vector<std::string> &servers) : servers_(servers) {} |   template<typename... Args> SNTPComponent(Args... servers) : servers_{servers...} {} | ||||||
|  |  | ||||||
|   // Note: set_servers() has been removed and replaced by a constructor - calling set_servers after setup would |  | ||||||
|   // have had no effect anyway, and making the strings immutable avoids the need to strdup their contents. |  | ||||||
|  |  | ||||||
|   void setup() override; |   void setup() override; | ||||||
|   void dump_config() override; |   void dump_config() override; | ||||||
| @@ -29,7 +34,13 @@ class SNTPComponent : public time::RealTimeClock { | |||||||
|   void time_synced(); |   void time_synced(); | ||||||
|  |  | ||||||
|  protected: |  protected: | ||||||
|   std::vector<std::string> servers_; | #ifdef USE_ESP8266 | ||||||
|  |   // On ESP8266, store pointers to PROGMEM strings to save RAM | ||||||
|  |   std::array<PGM_P, SNTP_SERVER_COUNT> servers_; | ||||||
|  | #else | ||||||
|  |   // On other platforms, store regular const char pointers | ||||||
|  |   std::array<const char *, SNTP_SERVER_COUNT> servers_; | ||||||
|  | #endif | ||||||
|   bool has_time_{false}; |   bool has_time_{false}; | ||||||
|  |  | ||||||
| #if defined(USE_ESP32) | #if defined(USE_ESP32) | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ from esphome.const import ( | |||||||
|     PLATFORM_RTL87XX, |     PLATFORM_RTL87XX, | ||||||
| ) | ) | ||||||
| from esphome.core import CORE | from esphome.core import CORE | ||||||
|  | from esphome.cpp_generator import ProgmemAssignmentExpression | ||||||
|  |  | ||||||
| DEPENDENCIES = ["network"] | DEPENDENCIES = ["network"] | ||||||
| sntp_ns = cg.esphome_ns.namespace("sntp") | sntp_ns = cg.esphome_ns.namespace("sntp") | ||||||
| @@ -43,7 +44,30 @@ CONFIG_SCHEMA = cv.All( | |||||||
|  |  | ||||||
| async def to_code(config): | async def to_code(config): | ||||||
|     servers = config[CONF_SERVERS] |     servers = config[CONF_SERVERS] | ||||||
|     var = cg.new_Pvariable(config[CONF_ID], servers) |     server_count = len(servers) | ||||||
|  |  | ||||||
|  |     # Define server count at compile time | ||||||
|  |     cg.add_define("SNTP_SERVER_COUNT", server_count) | ||||||
|  |  | ||||||
|  |     # Generate PROGMEM strings for ESP8266, regular strings for other platforms | ||||||
|  |     if CORE.is_esp8266: | ||||||
|  |         # On ESP8266, use PROGMEM to store strings in flash | ||||||
|  |         server_vars = [] | ||||||
|  |         for i, server in enumerate(servers): | ||||||
|  |             var_name = f"{config[CONF_ID].id}_server_{i}" | ||||||
|  |             # Create PROGMEM string: static const char var_name[] PROGMEM = "server"; | ||||||
|  |             assignment = ProgmemAssignmentExpression( | ||||||
|  |                 "char", var_name, cg.safe_exp(server) | ||||||
|  |             ) | ||||||
|  |             cg.add(assignment) | ||||||
|  |             server_vars.append(cg.RawExpression(var_name)) | ||||||
|  |         # Pass PROGMEM string pointers to constructor using ArrayInitializer | ||||||
|  |         var = cg.new_Pvariable(config[CONF_ID], cg.ArrayInitializer(*server_vars)) | ||||||
|  |     else: | ||||||
|  |         # On other platforms, pass regular string literals to constructor | ||||||
|  |         var = cg.new_Pvariable( | ||||||
|  |             config[CONF_ID], cg.ArrayInitializer(*[cg.safe_exp(s) for s in servers]) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     await cg.register_component(var, config) |     await cg.register_component(var, config) | ||||||
|     await time_.register_time(var, config) |     await time_.register_time(var, config) | ||||||
|   | |||||||
| @@ -87,6 +87,7 @@ | |||||||
| #define USE_MDNS_STORE_SERVICES | #define USE_MDNS_STORE_SERVICES | ||||||
| #define MDNS_SERVICE_COUNT 3 | #define MDNS_SERVICE_COUNT 3 | ||||||
| #define MDNS_DYNAMIC_TXT_COUNT 3 | #define MDNS_DYNAMIC_TXT_COUNT 3 | ||||||
|  | #define SNTP_SERVER_COUNT 3 | ||||||
| #define USE_MEDIA_PLAYER | #define USE_MEDIA_PLAYER | ||||||
| #define USE_NEXTION_TFT_UPLOAD | #define USE_NEXTION_TFT_UPLOAD | ||||||
| #define USE_NUMBER | #define USE_NUMBER | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user