1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-17 23:12:45 +01:00
RIOT/sys/shell/cmds/sntp.c
Marian Buschsieweke c06335b71b
sys/shell: make cmds submodules
Previously `shell_commands` was a "catch-all" module that included
shell commands for each and every used module that has a shell
companion. Instead, the new `shell_cmds` module is now used to provide
shell commands as individually selectable submodules, e.g.
`cmd_gnrc_icmpv6_echo` now provides the ICMPv6 echo command (a.k.a.
ping).

To still have a "catch all" module to pull in shell commands of modules
already used, `shell_cmds_default` was introduced. `shell_commands`
depends now on `shell_cmds_default` for backward compatibility, but
has been deprecated. New apps should use `shell_cmds_default`
instead.

For a handful of shell commands individual selection was already
possible. Those modules now depend on the corresponding `cmd_%` module
and they have been deprecated.
2022-09-16 13:15:45 +02:00

107 lines
3.0 KiB
C

/*
* Copyright (C) 2016 Luminița Lăzărescu <cluminita.lazarescu@gmail.com>
* 2017 Freie Universität Berlin
*
* 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>
*/
#include <stdio.h>
#include <time.h>
#include "net/af.h"
#include "net/gnrc/netif.h"
#include "net/ipv6/addr.h"
#include "net/ntp_packet.h"
#include "net/sntp.h"
#include "shell.h"
#include "timex.h"
#define _DEFAULT_TIMEOUT (500000LU)
static void _usage(char *cmd)
{
printf("Usage: %s <server addr>[%%<interface>] [<timeout in us>]\n", cmd);
printf("default: timeout = %lu\n", _DEFAULT_TIMEOUT);
}
static int _ntpdate(int argc, char **argv)
{
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;
}
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
return 0;
}
SHELL_COMMAND(ntpdate, "synchronizes with a remote time server", _ntpdate);