/* * Copyright (C) 2018 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. */ /** * @{ * * @file * @author Martine Lenders * * This implementation oriented itself on the [version by Mike * Muuss](http://ftp.arl.army.mil/~mike/ping.html) which was published under * public domain. The state-handling and duplicate detection was inspired by the * ping version of [inetutils](://www.gnu.org/software/inetutils/), which was * published under GPLv3 */ #ifdef MODULE_GNRC_ICMPV6 #include #include #include #include #include "bitfield.h" #include "byteorder.h" #include "msg.h" #include "net/gnrc.h" #include "net/gnrc/icmpv6.h" #include "net/icmpv6.h" #include "net/ipv6.h" #include "net/utils.h" #include "sched.h" #include "shell.h" #include "timex.h" #include "unaligned.h" #include "utlist.h" #include "ztimer.h" #ifdef MODULE_LUID #include "luid.h" #endif #ifdef MODULE_GNRC_IPV6_NIB #include "net/gnrc/ipv6/nib/nc.h" #endif #define _SEND_NEXT_PING (0xEF48) #define _PING_FINISH (0xEF49) #define CKTAB_SIZE (64U * 8) /* 64 byte * 8 bit/byte */ #define DEFAULT_COUNT (3U) #define DEFAULT_DATALEN (sizeof(uint32_t)) #define DEFAULT_ID (0x53) #define DEFAULT_INTERVAL_USEC (1U * US_PER_SEC) #define DEFAULT_TIMEOUT_USEC (1U * US_PER_SEC) typedef struct { gnrc_netreg_entry_t netreg; ztimer_t sched_timer; msg_t sched_msg; ipv6_addr_t host; char *hostname; unsigned long num_sent, num_recv, num_rept; unsigned long long tsum; unsigned tmin, tmax; unsigned count; size_t datalen; BITFIELD(cktab, CKTAB_SIZE); uint32_t timeout; uint32_t interval; gnrc_netif_t *netif; uint16_t id; uint8_t hoplimit; } _ping_data_t; static void _usage(char *cmdname); static int _configure(int argc, char **argv, _ping_data_t *data); static void _pinger(_ping_data_t *data); static void _print_reply(_ping_data_t *data, gnrc_pktsnip_t *icmpv6, uint32_t now_us, ipv6_addr_t *from, unsigned hoplimit, gnrc_netif_hdr_t *netif_hdr); static void _handle_reply(_ping_data_t *data, gnrc_pktsnip_t *pkt, uint32_t now_us); static int _finish(_ping_data_t *data); static int _gnrc_icmpv6_ping(int argc, char **argv) { _ping_data_t data = { .netreg = GNRC_NETREG_ENTRY_INIT_PID(ICMPV6_ECHO_REP, thread_getpid()), .count = DEFAULT_COUNT, .tmin = UINT_MAX, .datalen = DEFAULT_DATALEN, .timeout = DEFAULT_TIMEOUT_USEC, .interval = DEFAULT_INTERVAL_USEC, .id = DEFAULT_ID, }; int res; if ((res = _configure(argc, argv, &data)) != 0) { return res; } gnrc_netreg_register(GNRC_NETTYPE_ICMPV6, &data.netreg); _pinger(&data); do { msg_t msg; msg_receive(&msg); switch (msg.type) { case GNRC_NETAPI_MSG_TYPE_RCV: { _handle_reply(&data, msg.content.ptr, ztimer_now(ZTIMER_USEC)); gnrc_pktbuf_release(msg.content.ptr); break; } case _SEND_NEXT_PING: _pinger(&data); break; case _PING_FINISH: goto finish; default: /* requeue wrong packets */ msg_send(&msg, thread_getpid()); break; } } while (data.num_recv < data.count); finish: ztimer_remove(ZTIMER_USEC, &data.sched_timer); res = _finish(&data); gnrc_netreg_unregister(GNRC_NETTYPE_ICMPV6, &data.netreg); while (msg_avail() > 0) { msg_t msg; /* remove all remaining messages (likely caused by duplicates) */ if ((msg_try_receive(&msg) > 0) && (msg.type == GNRC_NETAPI_MSG_TYPE_RCV) && (((gnrc_pktsnip_t *)msg.content.ptr)->type == GNRC_NETTYPE_ICMPV6)) { gnrc_pktbuf_release(msg.content.ptr); } else { /* requeue other packets */ msg_send(&msg, thread_getpid()); } } return res; } SHELL_COMMAND(ping, "Ping via ICMPv6", _gnrc_icmpv6_ping); static void _usage(char *cmdname) { printf("%s [-c ] [-h] [-i ] [-s ]\n", cmdname); puts(" [-t hoplimit] [-W ] [%]"); puts(" count: number of pings (default: 3)"); puts(" ms interval: wait interval milliseconds between sending " "(default: 1000)"); puts(" packetsize: number of bytes in echo payload; must be >= 4 to " "measure round trip time (default: 4)"); puts(" hoplimit: Set the IP time to life/hoplimit " "(default: interface config)"); puts(" ms timeout: Time to wait for a response in milliseconds " "(default: 1000). The option affects only timeout in absence " "of any responses, otherwise wait for two RTTs"); } static int _configure(int argc, char **argv, _ping_data_t *data) { char *cmdname = argv[0]; int res = 1; /* parse command line arguments */ for (int i = 1; i < argc; i++) { char *arg = argv[i]; if (arg[0] != '-') { data->hostname = arg; res = netutils_get_ipv6(&data->host, (netif_t **)&data->netif, arg); if (res) { break; } } else { switch (arg[1]) { case 'c': if (((i++) + 1) < argc) { data->count = atoi(argv[i]); if (data->count > 0) { continue; } } /* intentionally falls through */ case 'h': res = 1; continue; /* intentionally falls through */ case 'i': if ((++i) < argc) { data->interval = (uint32_t)atoi(argv[i]) * US_PER_MS; continue; } /* intentionally falls through */ case 's': if ((++i) < argc) { data->datalen = atoi(argv[i]); continue; } /* intentionally falls through */ case 't': if ((++i) < argc) { data->hoplimit = atoi(argv[i]); continue; } /* intentionally falls through */ case 'W': if ((++i) < argc) { data->timeout = (uint32_t)atoi(argv[i]) * US_PER_MS; if (data->timeout > 0) { continue; } } /* intentionally falls through */ default: res = 1; break; } } } if (res != 0) { _usage(cmdname); } data->id ^= (ztimer_now(ZTIMER_USEC) & UINT16_MAX); #ifdef MODULE_LUID luid_custom(&data->id, sizeof(data->id), data->id); #endif return res; } static void _fill_payload(uint8_t *buf, size_t len) { uint8_t i = 0; if (len >= sizeof(uint32_t)) { uint32_t now = ztimer_now(ZTIMER_USEC); memcpy(buf, &now, sizeof(now)); len -= sizeof(now); buf += sizeof(now); } while (len--) { *buf++ = i++; } } static bool _check_payload(const void *buf, size_t len, uint32_t now, uint32_t *triptime, uint16_t *corrupt) { uint8_t i = 0; const uint8_t *data = buf; if (len >= sizeof(uint32_t)) { *triptime = now - unaligned_get_u32(buf); len -= sizeof(uint32_t); data += sizeof(uint32_t); } while (len--) { if (*data++ != i++) { *corrupt = data - (uint8_t *)buf - 1; return true; } } return false; } static void _pinger(_ping_data_t *data) { gnrc_pktsnip_t *pkt, *tmp; ipv6_hdr_t *ipv6; uint32_t timer; uint8_t *databuf; /* schedule next event (next ping or finish) ASAP */ if ((data->num_sent + 1) < data->count) { /* didn't send all pings yet - schedule next in data->interval */ data->sched_msg.type = _SEND_NEXT_PING; timer = data->interval; } else { /* Wait for the last ping to come back. * data->timeout: wait for a response in milliseconds. * Affects only timeout in absence of any responses, * otherwise ping waits for two max RTTs. */ data->sched_msg.type = _PING_FINISH; timer = data->timeout; if (data->num_recv) { /* approx. 2*tmax, in seconds (2 RTT) */ timer = (data->tmax / (512UL * 1024UL)) * US_PER_SEC; if (timer == 0) { timer = 1U * US_PER_SEC; } } } ztimer_set_msg(ZTIMER_USEC, &data->sched_timer, timer, &data->sched_msg, thread_getpid()); bf_unset(data->cktab, (size_t)data->num_sent % CKTAB_SIZE); pkt = gnrc_icmpv6_echo_build(ICMPV6_ECHO_REQ, data->id, (uint16_t)data->num_sent++, NULL, data->datalen); if (pkt == NULL) { puts("error: packet buffer full"); return; } databuf = (uint8_t *)(pkt->data) + sizeof(icmpv6_echo_t); tmp = gnrc_ipv6_hdr_build(pkt, NULL, &data->host); if (tmp == NULL) { puts("error: packet buffer full"); goto error_exit; } pkt = tmp; ipv6 = pkt->data; /* if data->hoplimit is unset (i.e. 0) gnrc_ipv6 will select hop limit */ ipv6->hl = data->hoplimit; if (data->netif != NULL) { tmp = gnrc_netif_hdr_build(NULL, 0, NULL, 0); if (tmp == NULL) { puts("error: packet buffer full"); goto error_exit; } gnrc_netif_hdr_set_netif(tmp->data, data->netif); pkt = gnrc_pkt_prepend(pkt, tmp); } /* add TX timestamp & test data */ _fill_payload(databuf, data->datalen); if (!gnrc_netapi_dispatch_send(GNRC_NETTYPE_IPV6, GNRC_NETREG_DEMUX_CTX_ALL, pkt)) { puts("error: unable to send ICMPv6 echo request"); goto error_exit; } return; error_exit: gnrc_pktbuf_release(pkt); } static void _print_reply(_ping_data_t *data, gnrc_pktsnip_t *icmpv6, uint32_t now, ipv6_addr_t *from, unsigned hoplimit, gnrc_netif_hdr_t *netif_hdr) { icmpv6_echo_t *icmpv6_hdr = icmpv6->data; kernel_pid_t if_pid = netif_hdr ? netif_hdr->if_pid : KERNEL_PID_UNDEF; int16_t rssi = netif_hdr ? netif_hdr->rssi : GNRC_NETIF_HDR_NO_RSSI; int16_t truncated; /* check if payload size matches expectation */ truncated = (data->datalen + sizeof(icmpv6_echo_t)) - icmpv6->size; if (icmpv6_hdr->type == ICMPV6_ECHO_REP) { char from_str[IPV6_ADDR_MAX_STR_LEN]; const char *dupmsg = " (DUP!)"; uint32_t triptime = 0; uint16_t recv_seq; uint16_t corrupted; /* not our ping */ if (byteorder_ntohs(icmpv6_hdr->id) != data->id) { return; } if (!ipv6_addr_is_multicast(&data->host) && !ipv6_addr_equal(from, &data->host)) { return; } recv_seq = byteorder_ntohs(icmpv6_hdr->seq); ipv6_addr_to_str(&from_str[0], from, sizeof(from_str)); if (gnrc_netif_highlander() || (if_pid == KERNEL_PID_UNDEF) || !ipv6_addr_is_link_local(from)) { printf("%u bytes from %s: icmp_seq=%u ttl=%u", (unsigned)icmpv6->size, from_str, recv_seq, hoplimit); } else { printf("%u bytes from %s%%%u: icmp_seq=%u ttl=%u", (unsigned)icmpv6->size, from_str, if_pid, recv_seq, hoplimit); } /* check if payload size matches */ if (truncated) { printf(" truncated by %d byte", truncated); } /* check response for corruption */ else if (_check_payload(icmpv6_hdr + 1, data->datalen, now, &triptime, &corrupted)) { printf(" corrupted at offset %u", corrupted); } if (rssi != GNRC_NETIF_HDR_NO_RSSI) { printf(" rssi=%"PRId16" dBm", rssi); } /* we can only calculate RTT (triptime) if payload was large enough for a TX timestamp */ if (data->datalen >= sizeof(uint32_t)) { printf(" time=%lu.%03lu ms", (long unsigned)triptime / 1000, (long unsigned)triptime % 1000); data->tsum += triptime; if (triptime < data->tmin) { data->tmin = triptime; } if (triptime > data->tmax) { data->tmax = triptime; } } if (bf_isset(data->cktab, recv_seq % CKTAB_SIZE)) { data->num_rept++; } else { bf_set(data->cktab, recv_seq % CKTAB_SIZE); data->num_recv++; dupmsg += 7; } puts(dupmsg); } } static void _handle_reply(_ping_data_t *data, gnrc_pktsnip_t *pkt, uint32_t now) { gnrc_pktsnip_t *netif, *ipv6, *icmpv6; gnrc_netif_hdr_t *netif_hdr; ipv6_hdr_t *ipv6_hdr; netif = gnrc_pktsnip_search_type(pkt, GNRC_NETTYPE_NETIF); ipv6 = gnrc_pktsnip_search_type(pkt, GNRC_NETTYPE_IPV6); icmpv6 = gnrc_pktsnip_search_type(pkt, GNRC_NETTYPE_ICMPV6); if ((ipv6 == NULL) || (icmpv6 == NULL)) { puts("No IPv6 or ICMPv6 header found in reply"); return; } ipv6_hdr = ipv6->data; netif_hdr = netif ? netif->data : NULL; _print_reply(data, icmpv6, now, &ipv6_hdr->src, ipv6_hdr->hl, netif_hdr); #ifdef MODULE_GNRC_IPV6_NIB /* successful ping to neighbor (NIB handles case if ipv6->src is not a * neighbor) can be taken as upper-layer hint for reachability: * https://tools.ietf.org/html/rfc4861#section-7.3.1 */ gnrc_ipv6_nib_nc_mark_reachable(&ipv6_hdr->src); #endif } static int _finish(_ping_data_t *data) { unsigned long tmp, nrecv, ndup; tmp = data->num_sent; nrecv = data->num_recv; ndup = data->num_rept; printf("\n--- %s PING statistics ---\n" "%lu packets transmitted, " "%lu packets received, ", data->hostname, tmp, nrecv); if (ndup) { printf("%lu duplicates, ", ndup); } if (tmp > 0) { tmp = ((tmp - nrecv) * 100) / tmp; } printf("%lu%% packet loss\n", tmp); if (data->tmin != UINT_MAX) { unsigned tavg = data->tsum / (nrecv + ndup); printf("round-trip min/avg/max = %u.%03u/%u.%03u/%u.%03u ms\n", data->tmin / 1000, data->tmin % 1000, tavg / 1000, tavg % 1000, data->tmax / 1000, data->tmax % 1000); } /* if condition is true, exit with 1 -- 'failure' */ return (nrecv == 0); } #endif /* MODULE_GNRC_ICMPV6 */ /** @} */