Select one of the symbols to view example projects that use it.
 
Outline
#include <string.h>
#include <time.h>
#include <sys/time.h>
#include "esp_system.h"
#include "esp_event.h"
#include "esp_log.h"
#include "esp_attr.h"
#include "esp_sleep.h"
#include "nvs_flash.h"
#include "protocol_examples_common.h"
#include "esp_netif_sntp.h"
#include "lwip/ip_addr.h"
#include "esp_sntp.h"
TAG
#define INET6_ADDRSTRLEN
boot_count
time_sync_notification_cb(struct timeval *)
app_main()
print_servers()
obtain_time()
Files
loading (1/4)...
SourceVuESP-IDF Framework and Examplessntp samplemain/sntp_example_main.c
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
/* LwIP SNTP example This example code is in the Public Domain (or CC0 licensed, at your option.) Unless required by applicable law or agreed to in writing, this software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. *//* ... */ #include <string.h> #include <time.h> #include <sys/time.h> #include "esp_system.h" #include "esp_event.h" #include "esp_log.h" #include "esp_attr.h" #include "esp_sleep.h" #include "nvs_flash.h" #include "protocol_examples_common.h" #include "esp_netif_sntp.h" #include "lwip/ip_addr.h" #include "esp_sntp.h"13 includes static const char *TAG = "example"; #ifndef INET6_ADDRSTRLEN #define INET6_ADDRSTRLEN 48 #endif /* Variable holding number of times ESP32 restarted since first boot. * It is placed into RTC memory using RTC_DATA_ATTR and * maintains its value when ESP32 wakes from deep sleep. *//* ... */ RTC_DATA_ATTR static int boot_count = 0; static void obtain_time(void); #ifdef CONFIG_SNTP_TIME_SYNC_METHOD_CUSTOM void sntp_sync_time(struct timeval *tv) { settimeofday(tv, NULL); ESP_LOGI(TAG, "Time is synchronized from custom code"); sntp_set_sync_status(SNTP_SYNC_STATUS_COMPLETED); }{...} /* ... */#endif void time_sync_notification_cb(struct timeval *tv) { ESP_LOGI(TAG, "Notification of a time synchronization event"); }{ ... } void app_main(void) { ++boot_count; ESP_LOGI(TAG, "Boot count: %d", boot_count); time_t now; struct tm timeinfo; time(&now); localtime_r(&now, &timeinfo); // Is time set? If not, tm_year will be (1970 - 1900). if (timeinfo.tm_year < (2016 - 1900)) { ESP_LOGI(TAG, "Time is not set yet. Connecting to WiFi and getting time over NTP."); obtain_time(); // update 'now' variable with current time time(&now); }{...} #ifdef CONFIG_SNTP_TIME_SYNC_METHOD_SMOOTH else { // add 500 ms error to the current system time. // Only to demonstrate a work of adjusting method! { ESP_LOGI(TAG, "Add a error for test adjtime"); struct timeval tv_now; gettimeofday(&tv_now, NULL); int64_t cpu_time = (int64_t)tv_now.tv_sec * 1000000L + (int64_t)tv_now.tv_usec; int64_t error_time = cpu_time + 500 * 1000L; struct timeval tv_error = { .tv_sec = error_time / 1000000L, .tv_usec = error_time % 1000000L }; settimeofday(&tv_error, NULL); } ESP_LOGI(TAG, "Time was set, now just adjusting it. Use SMOOTH SYNC method."); obtain_time(); // update 'now' variable with current time time(&now); } #endif char strftime_buf[64]; // Set timezone to Eastern Standard Time and print local time setenv("TZ", "EST5EDT,M3.2.0/2,M11.1.0", 1); tzset(); localtime_r(&now, &timeinfo); strftime(strftime_buf, sizeof(strftime_buf), "%c", &timeinfo); ESP_LOGI(TAG, "The current date/time in New York is: %s", strftime_buf); // Set timezone to China Standard Time setenv("TZ", "CST-8", 1); tzset(); localtime_r(&now, &timeinfo); strftime(strftime_buf, sizeof(strftime_buf), "%c", &timeinfo); ESP_LOGI(TAG, "The current date/time in Shanghai is: %s", strftime_buf); if (sntp_get_sync_mode() == SNTP_SYNC_MODE_SMOOTH) { struct timeval outdelta; while (sntp_get_sync_status() == SNTP_SYNC_STATUS_IN_PROGRESS) { adjtime(NULL, &outdelta); ESP_LOGI(TAG, "Waiting for adjusting time ... outdelta = %jd sec: %li ms: %li us", (intmax_t)outdelta.tv_sec, outdelta.tv_usec/1000, outdelta.tv_usec%1000); vTaskDelay(2000 / portTICK_PERIOD_MS); } } const int deep_sleep_sec = 10; ESP_LOGI(TAG, "Entering deep sleep for %d seconds", deep_sleep_sec); esp_deep_sleep(1000000LL * deep_sleep_sec); } static void print_servers(void) { ESP_LOGI(TAG, "List of configured NTP servers:"); for (uint8_t i = 0; i < SNTP_MAX_SERVERS; ++i){ if (esp_sntp_getservername(i)){ ESP_LOGI(TAG, "server %d: %s", i, esp_sntp_getservername(i)); } else { // we have either IPv4 or IPv6 address, let's print it char buff[INET6_ADDRSTRLEN]; ip_addr_t const *ip = esp_sntp_getserver(i); if (ipaddr_ntoa_r(ip, buff, INET6_ADDRSTRLEN) != NULL) ESP_LOGI(TAG, "server %d: %s", i, buff); } } } static void obtain_time(void) { ESP_ERROR_CHECK( nvs_flash_init() ); ESP_ERROR_CHECK(esp_netif_init()); ESP_ERROR_CHECK( esp_event_loop_create_default() ); #if LWIP_DHCP_GET_NTP_SRV /** * NTP server address could be acquired via DHCP, * see following menuconfig options: * 'LWIP_DHCP_GET_NTP_SRV' - enable STNP over DHCP * 'LWIP_SNTP_DEBUG' - enable debugging messages * * NOTE: This call should be made BEFORE esp acquires IP address from DHCP, * otherwise NTP option would be rejected by default. */ ESP_LOGI(TAG, "Initializing SNTP"); esp_sntp_config_t config = ESP_NETIF_SNTP_DEFAULT_CONFIG(CONFIG_SNTP_TIME_SERVER); config.start = false; // start SNTP service explicitly (after connecting) config.server_from_dhcp = true; // accept NTP offers from DHCP server, if any (need to enable *before* connecting) config.renew_servers_after_new_IP = true; // let esp-netif update configured SNTP server(s) after receiving DHCP lease config.index_of_first_server = 1; // updates from server num 1, leaving server 0 (from DHCP) intact // configure the event on which we renew servers #ifdef CONFIG_EXAMPLE_CONNECT_WIFI config.ip_event_to_renew = IP_EVENT_STA_GOT_IP; #else config.ip_event_to_renew = IP_EVENT_ETH_GOT_IP; #endif config.sync_cb = time_sync_notification_cb; // only if we need the notification function esp_netif_sntp_init(&config); #endif /* LWIP_DHCP_GET_NTP_SRV */ /* This helper function configures Wi-Fi or Ethernet, as selected in menuconfig. * Read "Establishing Wi-Fi or Ethernet Connection" section in * examples/protocols/README.md for more information about this function. */ ESP_ERROR_CHECK(example_connect()); #if LWIP_DHCP_GET_NTP_SRV ESP_LOGI(TAG, "Starting SNTP"); esp_netif_sntp_start(); #if LWIP_IPV6 && SNTP_MAX_SERVERS > 2 /* This demonstrates using IPv6 address as an additional SNTP server * (statically assigned IPv6 address is also possible) */ ip_addr_t ip6; if (ipaddr_aton("2a01:3f7::1", &ip6)) { // ipv6 ntp source "ntp.netnod.se" esp_sntp_setserver(2, &ip6); } #endif /* LWIP_IPV6 */ #else ESP_LOGI(TAG, "Initializing and starting SNTP"); #if CONFIG_LWIP_SNTP_MAX_SERVERS > 1 /* This demonstrates configuring more than one server */ esp_sntp_config_t config = ESP_NETIF_SNTP_DEFAULT_CONFIG_MULTIPLE(2, ESP_SNTP_SERVER_LIST(CONFIG_SNTP_TIME_SERVER, "pool.ntp.org" ) ); #else /* * This is the basic default config with one server and starting the service */ esp_sntp_config_t config = ESP_NETIF_SNTP_DEFAULT_CONFIG(CONFIG_SNTP_TIME_SERVER); #endif config.sync_cb = time_sync_notification_cb; // Note: This is only needed if we want #ifdef CONFIG_SNTP_TIME_SYNC_METHOD_SMOOTH config.smooth_sync = true; #endif esp_netif_sntp_init(&config); #endif print_servers(); // wait for time to be set time_t now = 0; struct tm timeinfo = { 0 }; int retry = 0; const int retry_count = 15; while (esp_netif_sntp_sync_wait(2000 / portTICK_PERIOD_MS) == ESP_ERR_TIMEOUT && ++retry < retry_count) { ESP_LOGI(TAG, "Waiting for system time to be set... (%d/%d)", retry, retry_count); } time(&now); localtime_r(&now, &timeinfo); ESP_ERROR_CHECK( example_disconnect() ); esp_netif_sntp_deinit(); }
Details