From 229f49ad5dddcbcbd63b37605f00137bc2022b1f Mon Sep 17 00:00:00 2001 From: Martine Lenders <mlenders@inf.fu-berlin.de> Date: Wed, 1 Apr 2015 17:40:36 +0200 Subject: [PATCH] shell: add icmpv6_echo shell handler --- sys/shell/commands/Makefile | 3 + sys/shell/commands/sc_icmpv6_echo.c | 240 ++++++++++++++++++++++++++++ sys/shell/commands/shell_commands.c | 11 ++ 3 files changed, 254 insertions(+) create mode 100644 sys/shell/commands/sc_icmpv6_echo.c diff --git a/sys/shell/commands/Makefile b/sys/shell/commands/Makefile index d2d2dbf6e9..b64d70c46c 100644 --- a/sys/shell/commands/Makefile +++ b/sys/shell/commands/Makefile @@ -64,6 +64,9 @@ endif ifneq (,$(filter ng_ipv6_nc,$(USEMODULE))) SRC += sc_ipv6_nc.c endif +ifneq (,$(filter ng_icmpv6_echo vtimer,$(USEMODULE))) + SRC += sc_icmpv6_echo.c +endif # TODO # Conditional building not possible at the moment due to diff --git a/sys/shell/commands/sc_icmpv6_echo.c b/sys/shell/commands/sc_icmpv6_echo.c new file mode 100644 index 0000000000..f9fb102576 --- /dev/null +++ b/sys/shell/commands/sc_icmpv6_echo.c @@ -0,0 +1,240 @@ +/* + * Copyright (C) 2015 Martine Lenders <mlenders@inf.fu-berlin.de> + * + * 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 + * + * @author Martine Lenders <mlenders@inf.fu-berlin.de> + */ + +#include <stdio.h> +#include <stdlib.h> + +#ifdef MODULE_NG_ICMPV6 + +#include "byteorder.h" +#include "net/ng_icmpv6.h" +#include "net/ng_ipv6/addr.h" +#include "net/ng_ipv6/hdr.h" +#include "net/ng_netbase.h" +#include "thread.h" +#include "utlist.h" +#include "vtimer.h" + +static uint16_t id = 0x53; +static uint16_t seq = 1; +static char ipv6_str[NG_IPV6_ADDR_MAX_STR_LEN]; + +static void usage(char **argv) +{ + printf("%s [<n>] <ipv6 addr> [<payload_len>]\n", argv[0]); +} + +void _set_payload(ng_icmpv6_echo_t *hdr, size_t payload_len) +{ + size_t i = 0; + uint8_t *payload = (uint8_t *)(hdr + 1); + + while (i < payload_len) { + if (seq > 0xff) { + payload[i] = (uint8_t)(seq >> 8); + payload[i + 1] = (uint8_t)(seq & 0xff); + i += 2; + } + else { + payload[i++] = (uint8_t)(seq & 0xff); + } + } + + if (i > payload_len) { + payload[payload_len - 1] = seq >> 8; + } +} + +int _handle_reply(ng_pktsnip_t *pkt, uint64_t time) +{ + ng_pktsnip_t *ipv6, *icmpv6; + ng_ipv6_hdr_t *ipv6_hdr; + ng_icmpv6_echo_t *icmpv6_hdr; + + LL_SEARCH_SCALAR(pkt, ipv6, type, NG_NETTYPE_IPV6); + LL_SEARCH_SCALAR(pkt, icmpv6, type, NG_NETTYPE_ICMPV6); + + if ((ipv6 == NULL) || (icmpv6 == NULL)) { + puts("error: IPv6 header or ICMPv6 header not found in reply"); + return 0; + } + + ipv6_hdr = ipv6->data; + icmpv6_hdr = icmpv6->data; + + if ((byteorder_ntohs(icmpv6_hdr->id) == id) && + (byteorder_ntohs(icmpv6_hdr->seq) == seq)) { + printf("%zu bytes from %s: id=%" PRIu16 " seq=%" PRIu16 " hop limit=%" PRIu8 + "time = %" PRIu64 ".%03" PRIu64 " ms\n", icmpv6->size, + ng_ipv6_addr_to_str(ipv6_str, &(ipv6_hdr->src), sizeof(ipv6_str)), + byteorder_ntohs(icmpv6_hdr->id), byteorder_ntohs(icmpv6_hdr->seq), + ipv6_hdr->hl, time / MS_IN_USEC, time % MS_IN_USEC); + } + else { + puts("error: unexpected parameters"); + return 0; + } + + return 1; +} + +int _icmpv6_ping(int argc, char **argv) +{ + int n = 3, success = 0, count; + size_t payload_len = 4; + char *addr_str; + ng_ipv6_addr_t addr; + ng_netreg_entry_t *ipv6_entry, my_entry = { NULL, NG_ICMPV6_ECHO_REP, + thread_getpid() + }; + timex_t min_rtt = { UINT32_MAX, UINT32_MAX }, max_rtt = { 0, 0 }; + timex_t sum_rtt = { 0, 0 }; + + switch (argc) { + case 1: + usage(argv); + return 1; + + case 2: + addr_str = argv[1]; + break; + + case 3: + n = atoi(argv[1]); + addr_str = argv[2]; + break; + + case 4: + default: + n = atoi(argv[1]); + addr_str = argv[2]; + payload_len = atoi(argv[3]); + break; + } + + if (ng_ipv6_addr_from_str(&addr, addr_str) == NULL) { + usage(argv); + return 1; + } + + if (ng_netreg_register(NG_NETTYPE_ICMPV6, &my_entry) < 0) { + puts("error: network registry is full"); + return 1; + } + + ipv6_entry = ng_netreg_lookup(NG_NETTYPE_IPV6, NG_NETREG_DEMUX_CTX_ALL); + + if (ipv6_entry == NULL) { + puts("error: ipv6 thread missing"); + return 1; + } + + count = n; + + while ((count--) > 0) { + msg_t msg; + ng_pktsnip_t *pkt; + timex_t start, stop, timeout = { 5, 0 }; + + pkt = ng_icmpv6_echo_req_build(id, seq, NULL, payload_len); + + if (pkt == NULL) { + puts("error: packet buffer full"); + return 1; + } + + _set_payload(pkt->data, payload_len); + + pkt = ng_netreg_hdr_build(NG_NETTYPE_IPV6, pkt, NULL, 0, addr.u8, + sizeof(ng_ipv6_addr_t)); + + if (pkt == NULL) { + puts("error: packet buffer full"); + return 1; + } + + vtimer_now(&start); + ng_netapi_send(ipv6_entry->pid, pkt); + + if (vtimer_msg_receive_timeout(&msg, timeout) >= 0) { + switch (msg.type) { + case NG_NETAPI_MSG_TYPE_RCV: + vtimer_now(&stop); + stop = timex_sub(stop, start); + success += _handle_reply((ng_pktsnip_t *)msg.content.ptr, + timex_uint64(stop)); + + if (timex_cmp(stop, max_rtt) > 0) { + max_rtt = stop; + } + + if (timex_cmp(stop, min_rtt) < 1) { + min_rtt = stop; + } + + sum_rtt = timex_add(sum_rtt, stop); + + break; + + default: + /* requeue wrong packets */ + msg_send(&msg, thread_getpid()); + break; + } + } + else { + puts("ping timeout"); + } + + seq++; + } + + seq = 1; + id++; + + ng_netreg_unregister(NG_NETTYPE_ICMPV6, &my_entry); + + printf("--- %s ping statistics ---\n", addr_str); + + if (success > 0) { + printf("%d packets transmitted, %d received, %d%% packet loss, time %" + PRIu64 " ms\n", n, success, (success - n) / n, + timex_uint64(sum_rtt) / MS_IN_USEC); + uint64_t avg_rtt = timex_uint64(sum_rtt) / n; /* get average */ + printf("rtt min/avg/max = " + "%" PRIu64 ".%03" PRIu64 "/" + "%" PRIu64 ".%03" PRIu64 "/" + "%" PRIu64 ".%03" PRIu64 " ms\n", + timex_uint64(min_rtt) / MS_IN_USEC, + timex_uint64(min_rtt) % MS_IN_USEC, + avg_rtt / MS_IN_USEC, avg_rtt % MS_IN_USEC, /* sum is now avg, see above */ + timex_uint64(max_rtt) / MS_IN_USEC, + timex_uint64(max_rtt) % MS_IN_USEC); + } + else { + printf("%d packets transmitted, 0 received, 100%% packet loss\n", n); + return 1; + } + + return 0; +} + +#endif + +/** + * @} + */ diff --git a/sys/shell/commands/shell_commands.c b/sys/shell/commands/shell_commands.c index 87ec5baa6d..c026933d83 100644 --- a/sys/shell/commands/shell_commands.c +++ b/sys/shell/commands/shell_commands.c @@ -145,6 +145,12 @@ extern int _read_sector(int argc, char **argv); extern int _read_bytes(int argc, char **argv); #endif +#ifdef MODULE_NG_ICMPV6_ECHO +#ifdef MODULE_VTIMER +extern int _icmpv6_ping(int argc, char **argv); +#endif +#endif + #ifdef MODULE_RANDOM extern int _mersenne_init(int argc, char **argv); extern int _mersenne_get(int argc, char **argv); @@ -250,6 +256,11 @@ const shell_command_t _shell_command_list[] = { {DISK_GET_SECTOR_COUNT, "Get the sector count of inserted memory card", _get_sectorcount}, {DISK_GET_BLOCK_SIZE, "Get the block size of inserted memory card", _get_blocksize}, #endif +#ifdef MODULE_NG_ICMPV6_ECHO +#ifdef MODULE_VTIMER + { "ping6", "Ping via ICMPv6", _icmpv6_ping }, +#endif +#endif #ifdef MODULE_RANDOM { "mersenne_init", "initializes the PRNG", _mersenne_init }, { "mersenne_get", "returns 32 bit of pseudo randomness", _mersenne_get }, -- GitLab