2016-05-19 11:42:01 +02:00
|
|
|
/** @file
|
|
|
|
* @brief ICMPv6 related functions
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2016 Intel Corporation
|
|
|
|
*
|
2017-01-19 02:01:01 +01:00
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
2016-05-19 11:42:01 +02:00
|
|
|
*/
|
|
|
|
|
2018-11-30 11:54:56 +01:00
|
|
|
#include <logging/log.h>
|
|
|
|
LOG_MODULE_REGISTER(net_icmpv6, CONFIG_NET_ICMPV6_LOG_LEVEL);
|
2016-05-19 11:42:01 +02:00
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <misc/slist.h>
|
2016-10-24 12:17:31 +02:00
|
|
|
#include <misc/byteorder.h>
|
2016-05-19 11:42:01 +02:00
|
|
|
#include <net/net_core.h>
|
2017-04-03 17:14:35 +02:00
|
|
|
#include <net/net_pkt.h>
|
2016-05-19 11:42:01 +02:00
|
|
|
#include <net/net_if.h>
|
|
|
|
#include "net_private.h"
|
|
|
|
#include "icmpv6.h"
|
2016-11-03 13:44:44 +01:00
|
|
|
#include "ipv6.h"
|
2016-12-13 14:50:31 +01:00
|
|
|
#include "net_stats.h"
|
2016-05-19 11:42:01 +02:00
|
|
|
|
2017-04-05 08:37:44 +02:00
|
|
|
#define PKT_WAIT_TIME K_SECONDS(1)
|
2017-03-15 09:15:45 +01:00
|
|
|
|
2016-05-19 11:42:01 +02:00
|
|
|
static sys_slist_t handlers;
|
|
|
|
|
2017-03-29 14:53:59 +02:00
|
|
|
const char *net_icmpv6_type2str(int icmpv6_type)
|
|
|
|
{
|
|
|
|
switch (icmpv6_type) {
|
|
|
|
case NET_ICMPV6_DST_UNREACH:
|
|
|
|
return "Destination Unreachable";
|
|
|
|
case NET_ICMPV6_PACKET_TOO_BIG:
|
|
|
|
return "Packet Too Big";
|
|
|
|
case NET_ICMPV6_TIME_EXCEEDED:
|
|
|
|
return "Time Exceeded";
|
|
|
|
case NET_ICMPV6_PARAM_PROBLEM:
|
|
|
|
return "IPv6 Bad Header";
|
|
|
|
case NET_ICMPV6_ECHO_REQUEST:
|
|
|
|
return "Echo Request";
|
|
|
|
case NET_ICMPV6_ECHO_REPLY:
|
|
|
|
return "Echo Reply";
|
|
|
|
case NET_ICMPV6_MLD_QUERY:
|
|
|
|
return "Multicast Listener Query";
|
|
|
|
case NET_ICMPV6_RS:
|
|
|
|
return "Router Solicitation";
|
|
|
|
case NET_ICMPV6_RA:
|
|
|
|
return "Router Advertisement";
|
|
|
|
case NET_ICMPV6_NS:
|
|
|
|
return "Neighbor Solicitation";
|
|
|
|
case NET_ICMPV6_NA:
|
|
|
|
return "Neighbor Advertisement";
|
|
|
|
case NET_ICMPV6_MLDv2:
|
|
|
|
return "Multicast Listener Report v2";
|
|
|
|
}
|
|
|
|
|
|
|
|
return "?";
|
|
|
|
}
|
|
|
|
|
2016-05-19 11:42:01 +02:00
|
|
|
void net_icmpv6_register_handler(struct net_icmpv6_handler *handler)
|
|
|
|
{
|
|
|
|
sys_slist_prepend(&handlers, &handler->node);
|
|
|
|
}
|
|
|
|
|
2017-03-03 13:15:43 +01:00
|
|
|
void net_icmpv6_unregister_handler(struct net_icmpv6_handler *handler)
|
|
|
|
{
|
|
|
|
sys_slist_find_and_remove(&handlers, &handler->node);
|
|
|
|
}
|
|
|
|
|
2018-12-17 15:34:00 +01:00
|
|
|
int net_icmpv6_finalize(struct net_pkt *pkt)
|
|
|
|
{
|
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(icmp_access,
|
|
|
|
struct net_icmp_hdr);
|
|
|
|
struct net_icmp_hdr *icmp_hdr;
|
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
icmp_hdr = (struct net_icmp_hdr *)net_pkt_get_data(pkt, &icmp_access);
|
2018-12-17 15:34:00 +01:00
|
|
|
if (!icmp_hdr) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
|
|
|
icmp_hdr->chksum = net_calc_chksum_icmpv6(pkt);
|
|
|
|
|
|
|
|
return net_pkt_set_data(pkt, &icmp_access);
|
|
|
|
}
|
|
|
|
|
2018-12-20 10:32:24 +01:00
|
|
|
int net_icmpv6_create(struct net_pkt *pkt, u8_t icmp_type, u8_t icmp_code)
|
2016-11-24 14:25:50 +01:00
|
|
|
{
|
2018-12-18 15:46:05 +01:00
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(icmp_access,
|
|
|
|
struct net_icmp_hdr);
|
|
|
|
struct net_icmp_hdr *icmp_hdr;
|
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
icmp_hdr = (struct net_icmp_hdr *)net_pkt_get_data(pkt, &icmp_access);
|
2018-12-18 15:46:05 +01:00
|
|
|
if (!icmp_hdr) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
|
|
|
icmp_hdr->type = icmp_type;
|
|
|
|
icmp_hdr->code = icmp_code;
|
|
|
|
icmp_hdr->chksum = 0;
|
|
|
|
|
|
|
|
return net_pkt_set_data(pkt, &icmp_access);
|
|
|
|
}
|
|
|
|
|
2018-12-20 12:59:56 +01:00
|
|
|
static
|
|
|
|
enum net_verdict icmpv6_handle_echo_request(struct net_pkt *pkt,
|
|
|
|
struct net_ipv6_hdr *ip_hdr,
|
|
|
|
struct net_icmp_hdr *icmp_hdr)
|
2018-12-18 15:46:05 +01:00
|
|
|
{
|
|
|
|
struct net_pkt *reply = NULL;
|
|
|
|
const struct in6_addr *src;
|
2018-11-01 15:06:39 +01:00
|
|
|
s16_t payload_len;
|
2016-11-24 14:25:50 +01:00
|
|
|
|
2018-12-20 12:59:56 +01:00
|
|
|
ARG_UNUSED(icmp_hdr);
|
|
|
|
|
2018-08-16 13:32:31 +02:00
|
|
|
NET_DBG("Received Echo Request from %s to %s",
|
2018-12-18 15:28:40 +01:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&ip_hdr->src)),
|
|
|
|
log_strdup(net_sprint_ipv6_addr(&ip_hdr->dst)));
|
2016-05-19 11:42:01 +02:00
|
|
|
|
2018-12-18 15:28:40 +01:00
|
|
|
payload_len = ntohs(ip_hdr->len) -
|
2018-12-18 15:46:05 +01:00
|
|
|
net_pkt_ipv6_ext_len(pkt) - NET_ICMPH_LEN;
|
2018-11-01 15:06:39 +01:00
|
|
|
if (payload_len < NET_ICMPV6_UNUSED_LEN) {
|
|
|
|
/* No identifier or sequence number present */
|
2018-12-18 15:46:05 +01:00
|
|
|
goto drop;
|
2017-03-15 09:15:45 +01:00
|
|
|
}
|
2016-11-24 14:25:50 +01:00
|
|
|
|
2018-12-18 15:46:05 +01:00
|
|
|
reply = net_pkt_alloc_with_buffer(net_pkt_iface(pkt), payload_len,
|
|
|
|
AF_INET6, IPPROTO_ICMPV6,
|
|
|
|
PKT_WAIT_TIME);
|
|
|
|
if (!reply) {
|
|
|
|
NET_DBG("DROP: No buffer");
|
2016-11-24 14:25:50 +01:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-18 15:46:05 +01:00
|
|
|
if (net_ipv6_is_addr_mcast(&ip_hdr->dst)) {
|
|
|
|
src = net_if_ipv6_select_src_addr(net_pkt_iface(pkt),
|
|
|
|
&ip_hdr->dst);
|
2017-03-03 15:42:29 +01:00
|
|
|
} else {
|
2018-12-18 15:46:05 +01:00
|
|
|
src = &ip_hdr->dst;
|
2017-03-03 15:42:29 +01:00
|
|
|
}
|
|
|
|
|
2016-11-29 14:32:33 +01:00
|
|
|
/* We must not set the destination ll address here but trust
|
|
|
|
* that it is set properly using a value from neighbor cache.
|
2018-12-18 15:46:05 +01:00
|
|
|
* Same for source as it points to original pkt ll src address.
|
2016-11-29 14:32:33 +01:00
|
|
|
*/
|
2018-12-18 15:46:05 +01:00
|
|
|
net_pkt_lladdr_dst(reply)->addr = NULL;
|
|
|
|
net_pkt_lladdr_src(reply)->addr = NULL;
|
|
|
|
|
2019-02-20 08:58:08 +01:00
|
|
|
if (net_ipv6_create(reply, src, &ip_hdr->src)) {
|
2018-12-18 15:46:05 +01:00
|
|
|
NET_DBG("DROP: wrong buffer");
|
|
|
|
goto drop;
|
|
|
|
}
|
2016-11-24 14:25:50 +01:00
|
|
|
|
2018-12-20 10:32:24 +01:00
|
|
|
if (net_icmpv6_create(reply, NET_ICMPV6_ECHO_REPLY, 0) ||
|
2019-02-08 09:35:06 +01:00
|
|
|
net_pkt_copy(reply, pkt, payload_len)) {
|
2018-12-18 15:46:05 +01:00
|
|
|
NET_DBG("DROP: wrong buffer");
|
2018-03-16 09:35:11 +01:00
|
|
|
goto drop;
|
|
|
|
}
|
2018-08-09 10:12:09 +02:00
|
|
|
|
2018-12-18 15:46:05 +01:00
|
|
|
net_pkt_cursor_init(reply);
|
2019-02-01 21:29:39 +01:00
|
|
|
net_ipv6_finalize(reply, IPPROTO_ICMPV6);
|
2016-05-19 11:42:01 +02:00
|
|
|
|
2018-08-16 13:32:31 +02:00
|
|
|
NET_DBG("Sending Echo Reply from %s to %s",
|
2018-12-18 15:46:05 +01:00
|
|
|
log_strdup(net_sprint_ipv6_addr(src)),
|
|
|
|
log_strdup(net_sprint_ipv6_addr(&ip_hdr->src)));
|
2016-05-19 11:42:01 +02:00
|
|
|
|
2018-12-18 15:46:05 +01:00
|
|
|
if (net_send_data(reply) < 0) {
|
2016-11-24 14:25:50 +01:00
|
|
|
goto drop;
|
2016-05-19 11:42:01 +02:00
|
|
|
}
|
|
|
|
|
2018-12-18 15:46:05 +01:00
|
|
|
net_stats_update_icmp_sent(net_pkt_iface(reply));
|
|
|
|
|
|
|
|
net_pkt_unref(pkt);
|
2016-05-19 11:42:01 +02:00
|
|
|
|
2017-03-03 09:13:36 +01:00
|
|
|
return NET_OK;
|
2016-05-19 11:42:01 +02:00
|
|
|
|
2016-11-24 14:25:50 +01:00
|
|
|
drop:
|
2018-12-18 15:46:05 +01:00
|
|
|
if (reply) {
|
|
|
|
net_pkt_unref(reply);
|
|
|
|
}
|
2017-03-15 09:15:45 +01:00
|
|
|
|
2018-12-18 15:46:05 +01:00
|
|
|
net_stats_update_icmp_drop(net_pkt_iface(pkt));
|
2016-06-14 08:42:01 +02:00
|
|
|
|
2016-11-24 14:25:50 +01:00
|
|
|
return NET_DROP;
|
2016-06-14 08:42:01 +02:00
|
|
|
}
|
|
|
|
|
2017-04-21 16:27:50 +02:00
|
|
|
int net_icmpv6_send_error(struct net_pkt *orig, u8_t type, u8_t code,
|
|
|
|
u32_t param)
|
2016-06-14 08:42:01 +02:00
|
|
|
{
|
2018-12-19 13:10:40 +01:00
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(ipv6_access, struct net_ipv6_hdr);
|
2017-03-15 09:15:45 +01:00
|
|
|
int err = -EIO;
|
2018-12-19 13:10:40 +01:00
|
|
|
struct net_ipv6_hdr *ip_hdr;
|
|
|
|
const struct in6_addr *src;
|
|
|
|
struct net_pkt *pkt;
|
|
|
|
size_t copy_len;
|
|
|
|
|
|
|
|
net_pkt_cursor_init(orig);
|
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
ip_hdr = (struct net_ipv6_hdr *)net_pkt_get_data(orig, &ipv6_access);
|
2018-12-19 13:10:40 +01:00
|
|
|
if (!ip_hdr) {
|
|
|
|
goto drop_no_pkt;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ip_hdr->nexthdr == IPPROTO_ICMPV6) {
|
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(icmpv6_access,
|
|
|
|
struct net_icmp_hdr);
|
|
|
|
struct net_icmp_hdr *icmp_hdr;
|
2016-06-14 08:42:01 +02:00
|
|
|
|
2018-12-19 13:10:40 +01:00
|
|
|
net_pkt_acknowledge_data(orig, &ipv6_access);
|
2017-06-28 14:34:02 +02:00
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
icmp_hdr = (struct net_icmp_hdr *)net_pkt_get_data(
|
2018-12-19 13:10:40 +01:00
|
|
|
orig, &icmpv6_access);
|
|
|
|
if (!icmp_hdr || icmp_hdr->code < 128) {
|
2016-06-14 08:42:01 +02:00
|
|
|
/* We must not send ICMP errors back */
|
2017-03-15 09:15:45 +01:00
|
|
|
err = -EINVAL;
|
2017-04-05 08:37:44 +02:00
|
|
|
goto drop_no_pkt;
|
2016-06-14 08:42:01 +02:00
|
|
|
}
|
|
|
|
|
2018-12-19 13:10:40 +01:00
|
|
|
net_pkt_cursor_init(orig);
|
2017-03-15 09:15:45 +01:00
|
|
|
}
|
2016-06-14 08:42:01 +02:00
|
|
|
|
2018-12-19 13:10:40 +01:00
|
|
|
if (ip_hdr->nexthdr == IPPROTO_UDP) {
|
|
|
|
copy_len = sizeof(struct net_ipv6_hdr) +
|
2016-06-14 08:42:01 +02:00
|
|
|
sizeof(struct net_udp_hdr);
|
2018-12-19 13:10:40 +01:00
|
|
|
} else if (ip_hdr->nexthdr == IPPROTO_TCP) {
|
|
|
|
copy_len = sizeof(struct net_ipv6_hdr) +
|
2017-02-22 12:51:21 +01:00
|
|
|
sizeof(struct net_tcp_hdr);
|
2016-06-14 08:42:01 +02:00
|
|
|
} else {
|
2018-12-19 13:10:40 +01:00
|
|
|
copy_len = net_pkt_get_len(orig);
|
2016-06-14 08:42:01 +02:00
|
|
|
}
|
|
|
|
|
2018-12-19 13:10:40 +01:00
|
|
|
pkt = net_pkt_alloc_with_buffer(net_pkt_iface(orig),
|
|
|
|
copy_len + NET_ICMPV6_UNUSED_LEN,
|
|
|
|
AF_INET6, IPPROTO_ICMPV6,
|
|
|
|
PKT_WAIT_TIME);
|
|
|
|
if (!pkt) {
|
2017-03-15 09:15:45 +01:00
|
|
|
err = -ENOMEM;
|
2018-12-19 13:10:40 +01:00
|
|
|
goto drop_no_pkt;
|
2016-06-14 08:42:01 +02:00
|
|
|
}
|
|
|
|
|
2018-12-19 13:10:40 +01:00
|
|
|
if (net_ipv6_is_addr_mcast(&ip_hdr->dst)) {
|
|
|
|
src = net_if_ipv6_select_src_addr(net_pkt_iface(pkt),
|
|
|
|
&ip_hdr->dst);
|
|
|
|
} else {
|
|
|
|
src = &ip_hdr->dst;
|
|
|
|
}
|
2016-06-14 08:42:01 +02:00
|
|
|
|
2019-02-20 08:58:08 +01:00
|
|
|
if (net_ipv6_create(pkt, src, &ip_hdr->src) ||
|
2018-12-20 10:32:24 +01:00
|
|
|
net_icmpv6_create(pkt, type, code)) {
|
2018-12-19 13:10:40 +01:00
|
|
|
goto drop;
|
|
|
|
}
|
2016-06-14 08:42:01 +02:00
|
|
|
|
2016-10-24 12:17:31 +02:00
|
|
|
/* Depending on error option, we store the param into the ICMP message.
|
|
|
|
*/
|
|
|
|
if (type == NET_ICMPV6_PARAM_PROBLEM) {
|
2019-02-20 10:01:57 +01:00
|
|
|
err = net_pkt_write_be32(pkt, param);
|
2018-12-19 13:10:40 +01:00
|
|
|
} else {
|
|
|
|
err = net_pkt_memset(pkt, 0, NET_ICMPV6_UNUSED_LEN);
|
2016-10-24 12:17:31 +02:00
|
|
|
}
|
|
|
|
|
2018-12-19 13:10:40 +01:00
|
|
|
/* Allocator might not have been able to allocate all requested space,
|
|
|
|
* so let's copy as much as we can.
|
|
|
|
*/
|
|
|
|
copy_len = net_pkt_available_buffer(pkt);
|
2016-06-14 08:42:01 +02:00
|
|
|
|
2019-02-08 09:35:06 +01:00
|
|
|
if (err || net_pkt_copy(pkt, orig, copy_len)) {
|
2018-12-19 13:10:40 +01:00
|
|
|
goto drop;
|
2016-06-14 08:42:01 +02:00
|
|
|
}
|
|
|
|
|
2018-09-11 09:16:03 +02:00
|
|
|
net_pkt_lladdr_src(pkt)->addr = net_pkt_lladdr_dst(orig)->addr;
|
|
|
|
net_pkt_lladdr_src(pkt)->len = net_pkt_lladdr_dst(orig)->len;
|
|
|
|
net_pkt_lladdr_dst(pkt)->addr = net_pkt_lladdr_src(orig)->addr;
|
|
|
|
net_pkt_lladdr_dst(pkt)->len = net_pkt_lladdr_src(orig)->len;
|
2016-06-14 08:42:01 +02:00
|
|
|
|
2018-12-19 13:10:40 +01:00
|
|
|
net_pkt_cursor_init(pkt);
|
2019-02-01 21:29:39 +01:00
|
|
|
net_ipv6_finalize(pkt, IPPROTO_ICMPV6);
|
2016-06-14 08:42:01 +02:00
|
|
|
|
2018-08-16 13:32:31 +02:00
|
|
|
NET_DBG("Sending ICMPv6 Error Message type %d code %d param %d"
|
|
|
|
" from %s to %s", type, code, param,
|
2018-12-19 13:10:40 +01:00
|
|
|
log_strdup(net_sprint_ipv6_addr(src)),
|
|
|
|
log_strdup(net_sprint_ipv6_addr(&ip_hdr->src)));
|
2016-06-14 08:42:01 +02:00
|
|
|
|
2017-04-05 08:37:44 +02:00
|
|
|
if (net_send_data(pkt) >= 0) {
|
2018-12-19 13:10:40 +01:00
|
|
|
net_stats_update_icmp_sent(net_pkt_iface(pkt));
|
2017-03-15 11:40:51 +01:00
|
|
|
return 0;
|
2016-06-14 08:42:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
drop:
|
2017-04-05 08:37:44 +02:00
|
|
|
net_pkt_unref(pkt);
|
2017-03-15 09:15:45 +01:00
|
|
|
|
2017-04-05 08:37:44 +02:00
|
|
|
drop_no_pkt:
|
2018-12-19 13:10:40 +01:00
|
|
|
net_stats_update_icmp_drop(net_pkt_iface(orig));
|
2016-06-14 08:42:01 +02:00
|
|
|
|
2017-03-15 09:15:45 +01:00
|
|
|
return err;
|
2016-06-14 08:42:01 +02:00
|
|
|
}
|
|
|
|
|
2016-11-03 13:44:44 +01:00
|
|
|
int net_icmpv6_send_echo_request(struct net_if *iface,
|
|
|
|
struct in6_addr *dst,
|
2017-04-21 16:27:50 +02:00
|
|
|
u16_t identifier,
|
|
|
|
u16_t sequence)
|
2016-11-03 13:44:44 +01:00
|
|
|
{
|
2018-12-17 15:28:21 +01:00
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(icmpv6_access,
|
|
|
|
struct net_icmpv6_echo_req);
|
|
|
|
int ret = -ENOBUFS;
|
|
|
|
struct net_icmpv6_echo_req *echo_req;
|
2016-11-03 13:44:44 +01:00
|
|
|
const struct in6_addr *src;
|
2017-04-05 08:37:44 +02:00
|
|
|
struct net_pkt *pkt;
|
2016-11-03 13:44:44 +01:00
|
|
|
|
|
|
|
src = net_if_ipv6_select_src_addr(iface, dst);
|
|
|
|
|
2018-12-17 15:28:21 +01:00
|
|
|
pkt = net_pkt_alloc_with_buffer(iface,
|
|
|
|
sizeof(struct net_icmpv6_echo_req),
|
|
|
|
AF_INET6, IPPROTO_ICMPV6,
|
|
|
|
PKT_WAIT_TIME);
|
2018-07-25 14:44:30 +02:00
|
|
|
if (!pkt) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2016-11-03 13:44:44 +01:00
|
|
|
|
2019-02-20 08:58:08 +01:00
|
|
|
if (net_ipv6_create(pkt, src, dst) ||
|
2018-12-20 10:32:24 +01:00
|
|
|
net_icmpv6_create(pkt, NET_ICMPV6_ECHO_REQUEST, 0)) {
|
2018-07-25 14:44:30 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
2016-11-03 13:44:44 +01:00
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
echo_req = (struct net_icmpv6_echo_req *)net_pkt_get_data(
|
2018-12-17 15:28:21 +01:00
|
|
|
pkt, &icmpv6_access);
|
|
|
|
if (!echo_req) {
|
|
|
|
goto drop;
|
|
|
|
}
|
2016-11-03 13:44:44 +01:00
|
|
|
|
2018-12-17 15:28:21 +01:00
|
|
|
echo_req->identifier = htons(identifier);
|
|
|
|
echo_req->sequence = htons(sequence);
|
2016-11-03 13:44:44 +01:00
|
|
|
|
2018-12-17 15:28:21 +01:00
|
|
|
net_pkt_set_data(pkt, &icmpv6_access);
|
2016-11-03 13:44:44 +01:00
|
|
|
|
2019-02-19 13:52:32 +01:00
|
|
|
net_pkt_cursor_init(pkt);
|
2019-02-01 21:29:39 +01:00
|
|
|
net_ipv6_finalize(pkt, IPPROTO_ICMPV6);
|
2016-11-03 13:44:44 +01:00
|
|
|
|
2018-08-16 13:32:31 +02:00
|
|
|
NET_DBG("Sending ICMPv6 Echo Request type %d from %s to %s",
|
|
|
|
NET_ICMPV6_ECHO_REQUEST,
|
2018-12-17 15:28:21 +01:00
|
|
|
log_strdup(net_sprint_ipv6_addr(src)),
|
|
|
|
log_strdup(net_sprint_ipv6_addr(dst)));
|
2016-11-03 13:44:44 +01:00
|
|
|
|
2017-04-05 08:37:44 +02:00
|
|
|
if (net_send_data(pkt) >= 0) {
|
2018-03-27 10:31:31 +02:00
|
|
|
net_stats_update_icmp_sent(iface);
|
2016-11-03 13:44:44 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-12-17 15:28:21 +01:00
|
|
|
net_stats_update_icmp_drop(iface);
|
|
|
|
|
2018-07-25 14:44:30 +02:00
|
|
|
ret = -EIO;
|
|
|
|
|
2017-03-10 10:27:32 +01:00
|
|
|
drop:
|
2017-04-05 08:37:44 +02:00
|
|
|
net_pkt_unref(pkt);
|
2016-11-03 13:44:44 +01:00
|
|
|
|
2018-07-25 14:44:30 +02:00
|
|
|
return ret;
|
2016-11-03 13:44:44 +01:00
|
|
|
}
|
|
|
|
|
2018-12-18 15:28:40 +01:00
|
|
|
enum net_verdict net_icmpv6_input(struct net_pkt *pkt,
|
|
|
|
struct net_ipv6_hdr *ip_hdr)
|
2016-05-19 11:42:01 +02:00
|
|
|
{
|
2018-12-18 15:28:40 +01:00
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(icmp_access,
|
|
|
|
struct net_icmp_hdr);
|
|
|
|
struct net_icmp_hdr *icmp_hdr;
|
2016-05-19 11:42:01 +02:00
|
|
|
struct net_icmpv6_handler *cb;
|
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
icmp_hdr = (struct net_icmp_hdr *)net_pkt_get_data(pkt, &icmp_access);
|
2018-12-18 15:28:40 +01:00
|
|
|
if (!icmp_hdr) {
|
|
|
|
NET_DBG("DROP: NULL ICMPv6 header");
|
|
|
|
return NET_DROP;
|
2018-12-04 12:10:16 +01:00
|
|
|
}
|
|
|
|
|
2018-12-18 15:28:40 +01:00
|
|
|
if (net_calc_chksum_icmpv6(pkt) != 0) {
|
|
|
|
NET_DBG("DROP: invalid checksum");
|
2018-12-11 10:38:05 +01:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-18 15:46:05 +01:00
|
|
|
net_pkt_acknowledge_data(pkt, &icmp_access);
|
|
|
|
|
2018-12-11 10:38:05 +01:00
|
|
|
NET_DBG("ICMPv6 %s received type %d code %d",
|
2018-12-18 15:28:40 +01:00
|
|
|
net_icmpv6_type2str(icmp_hdr->type),
|
|
|
|
icmp_hdr->type, icmp_hdr->code);
|
2018-12-11 10:38:05 +01:00
|
|
|
|
2018-03-27 10:31:31 +02:00
|
|
|
net_stats_update_icmp_recv(net_pkt_iface(pkt));
|
2017-03-15 10:02:26 +01:00
|
|
|
|
2017-02-08 15:08:23 +01:00
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&handlers, cb, node) {
|
2018-12-18 15:28:40 +01:00
|
|
|
if (cb->type == icmp_hdr->type &&
|
|
|
|
(cb->code == icmp_hdr->code || cb->code == 0)) {
|
2018-12-20 12:59:56 +01:00
|
|
|
return cb->handler(pkt, ip_hdr, icmp_hdr);
|
2016-05-19 11:42:01 +02:00
|
|
|
}
|
|
|
|
}
|
2018-12-04 12:10:16 +01:00
|
|
|
drop:
|
2018-03-27 10:31:31 +02:00
|
|
|
net_stats_update_icmp_drop(net_pkt_iface(pkt));
|
2017-03-15 10:02:26 +01:00
|
|
|
|
2016-05-19 11:42:01 +02:00
|
|
|
return NET_DROP;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct net_icmpv6_handler echo_request_handler = {
|
|
|
|
.type = NET_ICMPV6_ECHO_REQUEST,
|
|
|
|
.code = 0,
|
2018-12-18 15:46:05 +01:00
|
|
|
.handler = icmpv6_handle_echo_request,
|
2016-05-19 11:42:01 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
void net_icmpv6_init(void)
|
|
|
|
{
|
|
|
|
net_icmpv6_register_handler(&echo_request_handler);
|
|
|
|
}
|