mirror of
https://github.com/esphome/esphome.git
synced 2025-10-31 23:21:54 +00:00
Merge branch 'sntp_servers_flash' into integration
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