1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/sys/net/gnrc/network_layer/icmpv6/gnrc_icmpv6.c

168 lines
4.2 KiB
C
Raw Normal View History

2015-03-07 20:37:45 +01:00
/*
* 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 net_gnrc_icmpv6
2015-03-07 20:37:45 +01:00
* @{
*
* @file
*
* @author Martine Lenders <mlenders@inf.fu-berlin.de>
*/
2020-10-21 15:58:33 +02:00
#include <assert.h>
2015-03-07 20:37:45 +01:00
#include <errno.h>
#include <inttypes.h>
#include <stdlib.h>
#include "byteorder.h"
#include "net/ipv6/hdr.h"
2015-08-10 02:41:08 +02:00
#include "net/gnrc.h"
2017-05-22 15:52:11 +02:00
#include "net/gnrc/ipv6/nib.h"
#include "net/protnum.h"
2015-03-07 20:37:45 +01:00
#include "od.h"
#include "utlist.h"
#include "net/gnrc/icmpv6.h"
#include "net/gnrc/icmpv6/echo.h"
2015-03-07 20:37:45 +01:00
#define ENABLE_DEBUG (0)
#include "debug.h"
static inline uint16_t _calc_csum(gnrc_pktsnip_t *hdr,
gnrc_pktsnip_t *pseudo_hdr,
gnrc_pktsnip_t *payload)
2015-03-07 20:37:45 +01:00
{
uint16_t csum = 0;
uint16_t len = (uint16_t)hdr->size;
while (payload && (payload != hdr)) {
csum = inet_csum_slice(csum, payload->data, payload->size, len);
2015-03-07 20:37:45 +01:00
len += (uint16_t)payload->size;
payload = payload->next;
}
2015-08-07 14:56:36 +02:00
csum = inet_csum(csum, hdr->data, hdr->size);
csum = ipv6_hdr_inet_csum(csum, pseudo_hdr->data, PROTNUM_ICMPV6, len);
2015-03-07 20:37:45 +01:00
return ~csum;
}
void gnrc_icmpv6_demux(gnrc_netif_t *netif, gnrc_pktsnip_t *pkt)
2015-03-07 20:37:45 +01:00
{
gnrc_pktsnip_t *icmpv6, *ipv6;
icmpv6_hdr_t *hdr;
2015-03-07 20:37:45 +01:00
icmpv6 = gnrc_pktsnip_search_type(pkt, GNRC_NETTYPE_ICMPV6);
2015-03-07 20:37:45 +01:00
2015-07-22 14:38:28 +02:00
assert(icmpv6 != NULL);
2015-03-07 20:37:45 +01:00
/* there can be extension headers between IPv6 and ICMPv6 header so we have
* to search it */
ipv6 = gnrc_pktsnip_search_type(icmpv6, GNRC_NETTYPE_IPV6);
2015-03-07 20:37:45 +01:00
2015-07-22 14:38:28 +02:00
assert(ipv6 != NULL);
2016-03-11 04:44:14 +01:00
if (icmpv6->size < sizeof(icmpv6_hdr_t)) {
DEBUG("icmpv6: packet too short.\n");
gnrc_pktbuf_release(pkt);
2016-03-11 04:44:14 +01:00
return;
}
/* Note: size will be checked again in packet handlers */
2016-03-11 04:44:14 +01:00
hdr = (icmpv6_hdr_t *)icmpv6->data;
2015-03-07 20:37:45 +01:00
if (_calc_csum(icmpv6, ipv6, pkt)) {
DEBUG("icmpv6: wrong checksum.\n");
2019-01-24 14:57:36 +01:00
gnrc_pktbuf_release(pkt);
2015-03-07 20:37:45 +01:00
return;
}
switch (hdr->type) {
/* TODO: handle ICMPv6 errors */
#ifdef MODULE_GNRC_ICMPV6_ECHO
case ICMPV6_ECHO_REQ:
2015-03-07 20:37:45 +01:00
DEBUG("icmpv6: handle echo request.\n");
gnrc_icmpv6_echo_req_handle(netif, (ipv6_hdr_t *)ipv6->data,
(icmpv6_echo_t *)hdr, icmpv6->size);
2015-03-07 20:37:45 +01:00
break;
#endif
2017-05-22 15:52:11 +02:00
case ICMPV6_RTR_SOL:
case ICMPV6_RTR_ADV:
case ICMPV6_NBR_SOL:
case ICMPV6_NBR_ADV:
case ICMPV6_REDIRECT:
case ICMPV6_DAR:
case ICMPV6_DAC:
DEBUG("icmpv6: NDP message received. Handle with gnrc_ipv6_nib\n");
gnrc_ipv6_nib_handle_pkt(netif, ipv6->data, hdr, icmpv6->size);
2017-05-22 15:52:11 +02:00
break;
2015-03-07 20:37:45 +01:00
default:
2016-02-18 19:57:50 +01:00
DEBUG("icmpv6: unknown type field %u\n", hdr->type);
(void)netif;
2015-03-07 20:37:45 +01:00
break;
}
/* ICMPv6-all will be send in gnrc_ipv6.c so only dispatch of subtypes is
2015-03-07 20:37:45 +01:00
* needed */
if (!gnrc_netapi_dispatch_receive(GNRC_NETTYPE_ICMPV6, hdr->type, pkt)) {
DEBUG("icmpv6: no one interested in type %d\n", hdr->type);
gnrc_pktbuf_release(pkt);
2015-03-07 20:37:45 +01:00
}
}
2017-04-18 11:57:40 +02:00
gnrc_pktsnip_t *gnrc_icmpv6_build(gnrc_pktsnip_t *next, uint8_t type,
uint8_t code, size_t size)
2015-03-07 20:37:45 +01:00
{
gnrc_pktsnip_t *pkt;
icmpv6_hdr_t *icmpv6;
2015-03-07 20:37:45 +01:00
2017-04-18 11:57:40 +02:00
if ((pkt = gnrc_pktbuf_add(next, NULL, size, GNRC_NETTYPE_ICMPV6)) == NULL) {
DEBUG("icmpv6: no space left in packet buffer\n");
2015-03-07 20:37:45 +01:00
return NULL;
}
2016-02-18 19:57:50 +01:00
DEBUG("icmpv6: Building ICMPv6 message with type=%u, code=%u\n",
2015-03-07 20:37:45 +01:00
type, code);
icmpv6 = (icmpv6_hdr_t *)pkt->data;
2015-03-07 20:37:45 +01:00
icmpv6->type = type;
icmpv6->code = code;
icmpv6->csum.u16 = 0;
2015-03-07 20:37:45 +01:00
return pkt;
}
int gnrc_icmpv6_calc_csum(gnrc_pktsnip_t *hdr, gnrc_pktsnip_t *pseudo_hdr)
2015-03-07 20:37:45 +01:00
{
uint32_t csum = 0;
if (hdr == NULL) {
return -EFAULT;
}
if (hdr->type != GNRC_NETTYPE_ICMPV6) {
2015-03-07 20:37:45 +01:00
return -EBADMSG;
}
csum = _calc_csum(hdr, pseudo_hdr, hdr->next);
if (csum == 0) {
return -ENOENT;
}
((icmpv6_hdr_t *)hdr->data)->csum = byteorder_htons(csum);
2015-03-07 20:37:45 +01:00
return 0;
}
/**
* @}
*/