-
Notifications
You must be signed in to change notification settings - Fork 142
/
demo_netxduo_sntp_client.c
530 lines (385 loc) · 16.5 KB
/
demo_netxduo_sntp_client.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
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
/*
This is a small demo of the NetX SNTP Client on the high-performance NetX TCP/IP stack.
This demo relies on Thread, NetX and NetX SNTP Client API to execute the Simple Network Time
Protocol in unicast and broadcast modes.
*/
#include <stdio.h>
#include "nx_api.h"
#include "nx_ip.h"
#include "nxd_sntp_client.h"
/* Define SNTP packet size. */
#define NX_SNTP_CLIENT_PACKET_SIZE (NX_UDP_PACKET + 100)
/* Define SNTP packet pool size. */
#define NX_SNTP_CLIENT_PACKET_POOL_SIZE (4 * (NX_SNTP_CLIENT_PACKET_SIZE + sizeof(NX_PACKET)))
/* Define how often the demo checks for SNTP updates. */
#define DEMO_PERIODIC_CHECK_INTERVAL (1 * NX_IP_PERIODIC_RATE)
/* Define how often we check on SNTP server status. We expect updates from the SNTP
server about every hour using the SNTP Client defaults. For testing make this (much) shorter. */
#define CHECK_SNTP_UPDATES_TIMEOUT (180 * NX_IP_PERIODIC_RATE)
/* Set up generic network driver for demo program. */
void _nx_ram_network_driver(struct NX_IP_DRIVER_STRUCT *driver_req);
/* Application defined services of the NetX SNTP Client. */
UINT leap_second_handler(NX_SNTP_CLIENT *client_ptr, UINT leap_indicator);
UINT kiss_of_death_handler(NX_SNTP_CLIENT *client_ptr, UINT KOD_code);
VOID time_update_callback(NX_SNTP_TIME_MESSAGE *time_update_ptr, NX_SNTP_TIME *local_time);
/* Set up client thread and network resources. */
NX_PACKET_POOL client_packet_pool;
NX_IP client_ip;
TX_THREAD demo_client_thread;
NX_SNTP_CLIENT demo_sntp_client;
TX_EVENT_FLAGS_GROUP sntp_flags;
#define DEMO_SNTP_UPDATE_EVENT 1
/* Configure the SNTP Client to use IPv6. If not enabled, the
Client will use IPv4. Note: IPv6 must be enabled in NetX Duo
for the Client to communicate over IPv6. */
#ifdef FEATURE_NX_IPV6
/* #define USE_IPV6 */
#endif /* FEATURE_NX_IPV6 */
/* Configure the SNTP Client to use unicast SNTP. */
#define USE_UNICAST
#define CLIENT_IP_ADDRESS IP_ADDRESS(192,2,2,66)
#define SERVER_IP_ADDRESS IP_ADDRESS(192,2,2,92)
#define SERVER_IP_ADDRESS_2 SERVER_IP_ADDRESS
/* Set up the SNTP network and address index; */
UINT iface_index =0;
UINT prefix = 64;
UINT address_index;
/* Set up client thread entry point. */
void demo_client_thread_entry(ULONG info);
/* Define main entry point. */
int main()
{
/* Enter the ThreadX kernel. */
tx_kernel_enter();
return 0;
}
/* Define what the initial system looks like. */
void tx_application_define(void *first_unused_memory)
{
UINT status;
UCHAR *free_memory_pointer;
free_memory_pointer = (UCHAR *)first_unused_memory;
/* Create client packet pool. */
status = nx_packet_pool_create(&client_packet_pool, "SNTP Client Packet Pool",
NX_SNTP_CLIENT_PACKET_SIZE, free_memory_pointer,
NX_SNTP_CLIENT_PACKET_POOL_SIZE);
/* Check for errors. */
if (status != NX_SUCCESS)
{
return;
}
/* Initialize the NetX system. */
nx_system_initialize();
/* Update pointer to unallocated (free) memory. */
free_memory_pointer = free_memory_pointer + NX_SNTP_CLIENT_PACKET_POOL_SIZE;
/* Create Client IP instances */
status = nx_ip_create(&client_ip, "SNTP IP Instance", CLIENT_IP_ADDRESS,
0xFFFFFF00UL, &client_packet_pool, _nx_ram_network_driver,
free_memory_pointer, 2048, 1);
/* Check for error. */
if (status != NX_SUCCESS)
{
return;
}
free_memory_pointer = free_memory_pointer + 2048;
#ifndef NX_DISABLE_IPV4
/* Enable ARP and supply ARP cache memory. */
status = nx_arp_enable(&client_ip, (void **) free_memory_pointer, 2048);
/* Check for error. */
if (status != NX_SUCCESS)
{
return;
}
#endif /* NX_DISABLE_IPV4 */
/* Update pointer to unallocated (free) memory. */
free_memory_pointer = free_memory_pointer + 2048;
/* Enable UDP for client. */
status = nx_udp_enable(&client_ip);
/* Check for error. */
if (status != NX_SUCCESS)
{
return;
}
#ifndef NX_DISABLE_IPV4
status = nx_icmp_enable(&client_ip);
/* Check for error. */
if (status != NX_SUCCESS)
{
return;
}
#endif /* NX_DISABLE_IPV4 */
/* Create the client thread */
status = tx_thread_create(&demo_client_thread, "SNTP Client Thread", demo_client_thread_entry,
(ULONG)(&demo_sntp_client), free_memory_pointer, 2048,
4, 4, TX_NO_TIME_SLICE, TX_DONT_START);
/* Check for errors */
if (status != TX_SUCCESS)
{
return;
}
/* Create the event flags. */
status = tx_event_flags_create(&sntp_flags, "SNTP event flags");
/* Check for errors */
if (status != TX_SUCCESS)
{
return;
}
/* Update pointer to unallocated (free) memory. */
free_memory_pointer = free_memory_pointer + 2048;
/* set the SNTP network interface to the primary interface. */
iface_index = 0;
/* Create the SNTP Client to run in broadcast mode.. */
status = nx_sntp_client_create(&demo_sntp_client, &client_ip, iface_index, &client_packet_pool,
leap_second_handler,
kiss_of_death_handler,
NULL /* no random_number_generator callback */);
/* Check for error. */
if (status != NX_SUCCESS)
{
/* Bail out!*/
return;
}
tx_thread_resume(&demo_client_thread);
return;
}
/* Define size of buffer to display client's local time. */
#define BUFSIZE 50
/* Define the client thread. */
void demo_client_thread_entry(ULONG info)
{
UINT status;
UINT spin;
UINT server_status;
ULONG base_seconds;
ULONG base_fraction;
ULONG seconds, milliseconds, microseconds, fraction;
UINT wait = 0;
UINT error_counter = 0;
ULONG events = 0;
#ifdef USE_IPV6
NXD_ADDRESS sntp_server_address;
NXD_ADDRESS client_ip_address;
#endif
NX_PARAMETER_NOT_USED(info);
/* Give other threads (IP instance) initialize first. */
tx_thread_sleep(NX_IP_PERIODIC_RATE);
#ifdef USE_IPV6
/* Set up IPv6 services. */
status = nxd_ipv6_enable(&client_ip);
status += nxd_icmp_enable(&client_ip);
if (status != NX_SUCCESS)
return;
client_ip_address.nxd_ip_address.v6[0] = 0x20010db8;
client_ip_address.nxd_ip_address.v6[1] = 0x0000f101;
client_ip_address.nxd_ip_address.v6[2] = 0x0;
client_ip_address.nxd_ip_address.v6[3] = 0x101;
client_ip_address.nxd_ip_version = NX_IP_VERSION_V6;
/* Set the IPv6 server address. */
sntp_server_address.nxd_ip_address.v6[0] = 0x20010db8;
sntp_server_address.nxd_ip_address.v6[1] = 0x0000f101;
sntp_server_address.nxd_ip_address.v6[2] = 0x0;
sntp_server_address.nxd_ip_address.v6[3] = 0x00000106;
sntp_server_address.nxd_ip_version = NX_IP_VERSION_V6;
/* Establish the link local address for the host. The RAM driver creates
a virtual MAC address. */
status = nxd_ipv6_address_set(&client_ip, iface_index, NX_NULL, 10, NULL);
/* Check for link local address set error. */
if (status != NX_SUCCESS)
{
return;
}
/* Set the host global IP address. We are assuming a 64
bit prefix here but this can be any value (< 128). */
status = nxd_ipv6_address_set(&client_ip, iface_index, &client_ip_address, prefix, &address_index);
/* Check for global address set error. */
if (status != NX_SUCCESS)
{
return;
}
/* Wait while NetX Duo validates the global and link local addresses. */
tx_thread_sleep(5 * NX_IP_PERIODIC_RATE);
#endif
/* Setup time update callback function. */
nx_sntp_client_set_time_update_notify(&demo_sntp_client, time_update_callback);
/* Set up client time updates depending on mode. */
#ifdef USE_UNICAST
/* Initialize the Client for unicast mode to poll the SNTP server once an hour. */
#ifdef USE_IPV6
/* Use the duo service to set up the Client and set the IPv6 SNTP server. Note: this
can take either an IPv4 or IPv6 address. */
status = nxd_sntp_client_initialize_unicast(&demo_sntp_client, &sntp_server_address);
#else
/* Use the IPv4 service to set up the Client and set the IPv4 SNTP server. */
status = nx_sntp_client_initialize_unicast(&demo_sntp_client, SERVER_IP_ADDRESS);
#endif /* USE_IPV6 */
#else /* Broadcast mode */
/* Initialize the Client for broadcast mode, no roundtrip calculation required and a broadcast SNTP service. */
#ifdef USE_IPV6
/* Use the duo service to initialize the Client and set IPv6 SNTP all hosts multicast address.
(Note: This can take either an IPv4 or IPv6 address.)*/
status = nxd_sntp_client_initialize_broadcast(&demo_sntp_client, &sntp_server_address, NX_NULL);
#else
/* Use the IPv4 service to initialize the Client and set IPv4 SNTP broadcast address. */
status = nx_sntp_client_initialize_broadcast(&demo_sntp_client, NX_NULL, SERVER_IP_ADDRESS);
#endif /* USE_IPV6 */
#endif /* USE_UNICAST */
/* Check for error. */
if (status != NX_SUCCESS)
{
return;
}
/* Set the base time which is approximately the number of seconds since the turn of the last century.
If this is not available in SNTP format, the nx_sntp_client_utility_add_msecs_to_ntp_time service
can convert milliseconds to fraction. For how to compute NTP seconds from real time, read the
NetX SNTP User Guide.
Otherwise set the base time to zero and set NX_SNTP_CLIENT_IGNORE_MAX_ADJUST_STARTUP to NX_TRUE for
the SNTP CLient to accept the first time update without applying a minimum or maximum adjustment
parameters (NX_SNTP_CLIENT_MIN_TIME_ADJUSTMENT and NX_SNTP_CLIENT_MAX_TIME_ADJUSTMENT). */
base_seconds = 0xd2c96b90; /* Jan 24, 2012 UTC */
base_fraction = 0xa132db1e;
/* Apply to the SNTP Client local time. */
status = nx_sntp_client_set_local_time(&demo_sntp_client, base_seconds, base_fraction);
/* Check for error. */
if (status != NX_SUCCESS)
{
return;
}
/* Run whichever service the client is configured for. */
#ifdef USE_UNICAST
status = nx_sntp_client_run_unicast(&demo_sntp_client);
#else
status = nx_sntp_client_run_broadcast(&demo_sntp_client);
#endif /* USE_UNICAST */
if (status != NX_SUCCESS)
{
return;
}
spin = NX_TRUE;
/* Now check periodically for time changes. */
while(spin)
{
/* Wait for a server update event. */
tx_event_flags_get(&sntp_flags, DEMO_SNTP_UPDATE_EVENT, TX_OR_CLEAR, &events, DEMO_PERIODIC_CHECK_INTERVAL);
if (events == DEMO_SNTP_UPDATE_EVENT)
{
/* Check for valid SNTP server status. */
status = nx_sntp_client_receiving_updates(&demo_sntp_client, &server_status);
if ((status != NX_SUCCESS) || (server_status == NX_FALSE))
{
/* We do not have a valid update. Skip processing any time data. */
/* If this happens repeatedly, consider stopping the SNTP Client thread, picking another
SNTP server and resuming the SNTP Client thread task (more details about that in the
comments at the end of this function).
If SNTP Client configurable parameters are too restrictive, such as Max Adjustment, that may also cause
valid server updates to be rejected. Configurable parameters, however, cannot be changed at run time.*/
continue;
}
/* We have a valid update. Get the SNTP Client time. */
status = nx_sntp_client_get_local_time_extended(&demo_sntp_client, &seconds, &fraction, NX_NULL, 0);
/* Convert fraction to microseconds. */
nx_sntp_client_utility_fraction_to_usecs(fraction, µseconds);
milliseconds = ((microseconds + 500) / 1000);
if (status != NX_SUCCESS)
{
printf("Internal error with getting local time 0x%x\n", status);
error_counter++;
}
else
{
printf("\nSNTP updated\n");
printf("Time: %lu.%03lu sec.\r\n", seconds, milliseconds);
}
/* Clear all events in our event flag. */
events = 0;
}
else
{
/* No SNTP update event.
In the meantime, if we have an RTC we might want to check its notion of time.
In this demo, we simulate the passage of time on our 'RTC' really just the CPU counter,
assuming that seconds and milliseconds have previously been set to a base (starting) time
(as was the SNTP Client before running it) */
seconds += 1; /* This is the sleep time (1 second) so is pretty close to an RTC */
milliseconds += 1; /* We don't know this value but for demonstration purposes we change it */
/* Update our timer. */
wait += DEMO_PERIODIC_CHECK_INTERVAL;
/* Check if it is time to display the local 'RTC' time. */
if (wait >= CHECK_SNTP_UPDATES_TIMEOUT)
{
/* It is. Reset the timeout and print local time. */
wait = 0;
printf("Time: %lu.%03lu sec.\r\n", seconds, milliseconds);
}
}
}
/* We can stop the SNTP service if for example we think the SNTP server has stopped sending updates.
To restart the SNTP Client, simply call the nx_sntp_client_initialize_unicast or nx_sntp_client_initialize_broadcast
using another SNTP server IP address as input, and resume the SNTP Client by calling nx_sntp_client_run_unicast or
nx_sntp_client_run_braodcast. */
status = nx_sntp_client_stop(&demo_sntp_client);
if (status != NX_SUCCESS)
{
error_counter++;
}
/* When done with the SNTP Client, we delete it */
status = nx_sntp_client_delete(&demo_sntp_client);
return;
}
/* This application defined handler for handling an impending leap second is not
required by the SNTP Client. The default handler below only logs the event for
every time stamp received with the leap indicator set. */
UINT leap_second_handler(NX_SNTP_CLIENT *client_ptr, UINT leap_indicator)
{
NX_PARAMETER_NOT_USED(client_ptr);
NX_PARAMETER_NOT_USED(leap_indicator);
/* Handle the leap second handler... */
return NX_SUCCESS;
}
/* This application defined handler for handling a Kiss of Death packet is not
required by the SNTP Client. A KOD handler should determine
if the Client task should continue vs. abort sending/receiving time data
from its current time server, and if aborting if it should remove
the server from its active server list.
Note that the KOD list of codes is subject to change. The list
below is current at the time of this software release. */
UINT kiss_of_death_handler(NX_SNTP_CLIENT *client_ptr, UINT KOD_code)
{
UINT remove_server_from_list = NX_FALSE;
UINT status = NX_SUCCESS;
NX_PARAMETER_NOT_USED(client_ptr);
/* Handle kiss of death by code group. */
switch (KOD_code)
{
case NX_SNTP_KOD_RATE:
case NX_SNTP_KOD_NOT_INIT:
case NX_SNTP_KOD_STEP:
/* Find another server while this one is temporarily out of service. */
status = NX_SNTP_KOD_SERVER_NOT_AVAILABLE;
break;
case NX_SNTP_KOD_AUTH_FAIL:
case NX_SNTP_KOD_NO_KEY:
case NX_SNTP_KOD_CRYP_FAIL:
/* These indicate the server will not service client with time updates
without successful authentication. */
remove_server_from_list = NX_TRUE;
break;
default:
/* All other codes. Remove server before resuming time updates. */
remove_server_from_list = NX_TRUE;
break;
}
/* Removing the server from the active server list? */
if (remove_server_from_list)
{
/* Let the caller know it has to bail on this server before resuming service. */
status = NX_SNTP_KOD_REMOVE_SERVER;
}
return status;
}
/* This application defined handler for notifying SNTP time update event. */
VOID time_update_callback(NX_SNTP_TIME_MESSAGE *time_update_ptr, NX_SNTP_TIME *local_time)
{
NX_PARAMETER_NOT_USED(time_update_ptr);
NX_PARAMETER_NOT_USED(local_time);
tx_event_flags_set(&sntp_flags, DEMO_SNTP_UPDATE_EVENT, TX_OR);
}