1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/sys/shell/commands/sc_sntp.c

107 lines
3.0 KiB
C
Raw Normal View History

2016-10-29 16:59:00 +02:00
/*
* Copyright (C) 2016 Luminița Lăzărescu <cluminita.lazarescu@gmail.com>
* 2017 Freie Universität Berlin
2016-10-29 16:59:00 +02:00
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup sys_shell_commands
* @{
*
* @file
* @brief Prints the real time offset from the system time
*
* @author Luminița Lăzărescu <cluminita.lazarescu@gmail.com>
* @author Martine Lenders <m.lenders@fu-berlin.de>
2016-10-29 16:59:00 +02:00
*/
#include <stdio.h>
#include <time.h>
2016-10-29 16:59:00 +02:00
#include "net/af.h"
2022-02-25 18:32:47 +01:00
#include "net/gnrc/netif.h"
2016-10-29 16:59:00 +02:00
#include "net/ipv6/addr.h"
#include "net/ntp_packet.h"
#include "net/sntp.h"
#include "shell.h"
#include "timex.h"
2016-10-29 16:59:00 +02:00
#define _DEFAULT_TIMEOUT (500000LU)
2016-10-29 16:59:00 +02:00
static void _usage(char *cmd)
{
printf("Usage: %s <server addr>[%%<interface>] [<timeout in us>]\n", cmd);
printf("default: timeout = %lu\n", _DEFAULT_TIMEOUT);
2016-10-29 16:59:00 +02:00
}
static int _ntpdate(int argc, char **argv)
2016-10-29 16:59:00 +02:00
{
uint32_t timeout = _DEFAULT_TIMEOUT;
if (argc < 2) {
_usage(argv[0]);
return 1;
}
sock_udp_ep_t server = { .port = NTP_PORT, .family = AF_INET6 };
ipv6_addr_t *addr = (ipv6_addr_t *)&server.addr;
char *iface = ipv6_addr_split_iface(argv[1]);
kernel_pid_t src_iface = iface ? atoi(iface) : KERNEL_PID_UNDEF;
if (ipv6_addr_from_str(addr, argv[1]) == NULL) {
puts("error: malformed address");
return 1;
}
if (ipv6_addr_is_link_local(addr) || (src_iface != KERNEL_PID_UNDEF)) {
size_t ifnum = gnrc_netif_numof();
if (src_iface == KERNEL_PID_UNDEF) {
if (ifnum == 1) {
src_iface = gnrc_netif_iter(NULL)->pid;
}
else {
puts("error: link local target needs interface parameter (use \"<address>%<ifnum>\")\n");
return 1;
}
}
else {
if (gnrc_netif_get_by_pid(src_iface) == NULL) {
printf("error: %"PRIkernel_pid" is not a valid interface.\n", src_iface);
return 1;
}
}
server.netif = src_iface;
}
2016-10-29 16:59:00 +02:00
if (argc > 2) {
timeout = atoi(argv[2]);
}
if (sntp_sync(&server, timeout) < 0) {
puts("Error in synchronization");
return 1;
}
#if defined(MODULE_NEWLIB) || defined(MODULE_PICOLIBC)
struct tm *tm;
time_t time = (time_t)(sntp_get_unix_usec() / US_PER_SEC);
tm = gmtime(&time);
printf("%04i-%02i-%02i %02i:%02i:%02i UTC (%i us)\n",
tm->tm_year + 1900, tm->tm_mon + 1, tm->tm_mday,
tm->tm_hour, tm->tm_min, tm->tm_sec,
(int)sntp_get_offset());
#else
uint64_t time = sntp_get_unix_usec();
printf("%" PRIu32 ".%" PRIu32 " (%i us)\n",
(uint32_t)(time / US_PER_SEC),
(uint32_t)(time / US_PER_SEC),
(int)sntp_get_offset());
#endif
2016-10-29 16:59:00 +02:00
return 0;
}
SHELL_COMMAND(ntpdate, "synchronizes with a remote time server", _ntpdate);