mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
395 lines
12 KiB
C
395 lines
12 KiB
C
/*
|
|
* Copyright (C) 2014 Martine Lenders <mlenders@inf.fu-berlin.de>
|
|
* 2019 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
|
|
*/
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "embUnit.h"
|
|
|
|
#include "net/ipv6/addr.h"
|
|
#include "net/ipv6/hdr.h"
|
|
#include "net/ipv6/ext/frag.h"
|
|
#include "net/gnrc/pktbuf.h"
|
|
#include "net/protnum.h"
|
|
#include "net/inet_csum.h"
|
|
|
|
#include "unittests-constants.h"
|
|
#include "tests-ipv6_hdr.h"
|
|
|
|
#define OTHER_BYTE (TEST_UINT16 >> 8)
|
|
#define DEFAULT_TEST_SRC { { \
|
|
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, \
|
|
0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f \
|
|
} \
|
|
}
|
|
#define DEFAULT_TEST_DST { { \
|
|
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, \
|
|
0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f \
|
|
} \
|
|
}
|
|
|
|
static void test_ipv6_hdr_set_version(void)
|
|
{
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { TEST_UINT8 };
|
|
|
|
ipv6_hdr_set_version((ipv6_hdr_t *)val);
|
|
|
|
/*
|
|
* Header format:
|
|
* 1
|
|
* +----+--
|
|
* | 6 |
|
|
* +----+--
|
|
*/
|
|
TEST_ASSERT_EQUAL_INT(0x60, val[0] & 0xf0);
|
|
TEST_ASSERT_EQUAL_INT(TEST_UINT8 & 0x0f, val[0] & 0x0f);
|
|
}
|
|
|
|
static void test_ipv6_hdr_get_version(void)
|
|
{
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { TEST_UINT8 };
|
|
|
|
/*
|
|
* Header format:
|
|
* 8
|
|
* +----+----
|
|
* | 6 |
|
|
* +----+----
|
|
*/
|
|
TEST_ASSERT_EQUAL_INT(TEST_UINT8 >> 4, ipv6_hdr_get_version((ipv6_hdr_t *)val));
|
|
}
|
|
|
|
static void test_ipv6_hdr_is_ipv6_hdr__false(void)
|
|
{
|
|
/*
|
|
* Header format:
|
|
* 8
|
|
* +----+----
|
|
* | 6 |
|
|
* +----+----
|
|
*/
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { 0 };
|
|
|
|
TEST_ASSERT(!ipv6_hdr_is((ipv6_hdr_t *)val));
|
|
}
|
|
|
|
static void test_ipv6_hdr_is_ipv6_hdr__true(void)
|
|
{
|
|
/*
|
|
* Header format:
|
|
* 8
|
|
* +----+----
|
|
* | 6 |
|
|
* +----+----
|
|
*/
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { 0x60 | (TEST_UINT8 & 0x0f) };
|
|
|
|
TEST_ASSERT(ipv6_hdr_is((ipv6_hdr_t *)val));
|
|
}
|
|
|
|
static void test_ipv6_hdr_set_tc(void)
|
|
{
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { TEST_UINT8, 0 };
|
|
|
|
ipv6_hdr_set_tc((ipv6_hdr_t *)val, OTHER_BYTE);
|
|
|
|
/*
|
|
* Header format:
|
|
* 8
|
|
* +----+--------+--
|
|
* | 6 | tc |
|
|
* +----+--------+--
|
|
*/
|
|
TEST_ASSERT_EQUAL_INT((TEST_UINT8 & 0xf0) | (OTHER_BYTE >> 4), val[0]);
|
|
TEST_ASSERT_EQUAL_INT((OTHER_BYTE << 4) & 0xf0, val[1]);
|
|
}
|
|
|
|
static void test_ipv6_hdr_set_tc_ecn(void)
|
|
{
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { TEST_UINT8 };
|
|
|
|
ipv6_hdr_set_tc_ecn((ipv6_hdr_t *)val, OTHER_BYTE);
|
|
|
|
/*
|
|
* Header format:
|
|
* 8
|
|
* +----+--------+--
|
|
* | 6 | tc |
|
|
* +----+--------+--
|
|
* / \
|
|
* 8 12
|
|
* +----+------------+
|
|
* | ecn| dscp |
|
|
* +----+------------+
|
|
*/
|
|
TEST_ASSERT_EQUAL_INT((TEST_UINT8 & 0xf3) | ((OTHER_BYTE & 0x03) << 2), val[0]);
|
|
}
|
|
|
|
static void test_ipv6_hdr_set_tc_dscp(void)
|
|
{
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { TEST_UINT8, 0 };
|
|
|
|
ipv6_hdr_set_tc_dscp((ipv6_hdr_t *)val, OTHER_BYTE);
|
|
|
|
/*
|
|
* Header format:
|
|
* 8
|
|
* +----+--------+--
|
|
* | 6 | tc |
|
|
* +----+--------+--
|
|
* / \
|
|
* 8 12
|
|
* +----+------------+
|
|
* | ecn| dscp |
|
|
* +----+------------+
|
|
*/
|
|
TEST_ASSERT_EQUAL_INT((TEST_UINT8 & 0xfc) | ((OTHER_BYTE & 0x30) >> 4), val[0]);
|
|
TEST_ASSERT_EQUAL_INT((OTHER_BYTE & 0x0f) << 4, val[1]);
|
|
}
|
|
|
|
static void test_ipv6_hdr_get_tc(void)
|
|
{
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { TEST_UINT8, OTHER_BYTE };
|
|
|
|
/*
|
|
* Header format:
|
|
* 8
|
|
* +----+--------+--
|
|
* | 6 | tc |
|
|
* +----+--------+--
|
|
*/
|
|
TEST_ASSERT_EQUAL_INT(((TEST_UINT8 << 4) & 0xf0) | (OTHER_BYTE >> 4),
|
|
ipv6_hdr_get_tc((ipv6_hdr_t *)val));
|
|
}
|
|
|
|
static void test_ipv6_hdr_get_tc_ecn(void)
|
|
{
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { TEST_UINT8 };
|
|
|
|
/*
|
|
* Header format:
|
|
* 8
|
|
* +----+--------+--
|
|
* | 6 | tc |
|
|
* +----+--------+--
|
|
* / \
|
|
* 8 12
|
|
* +----+------------+
|
|
* | ecn| dscp |
|
|
* +----+------------+
|
|
*/
|
|
TEST_ASSERT_EQUAL_INT(TEST_UINT8 & 0x03, ipv6_hdr_get_tc_ecn((ipv6_hdr_t *)val));
|
|
}
|
|
|
|
static void test_ipv6_hdr_get_tc_dscp(void)
|
|
{
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { TEST_UINT8, OTHER_BYTE };
|
|
|
|
/*
|
|
* Header format:
|
|
* 8
|
|
* +----+--------+--
|
|
* | 6 | tc |
|
|
* +----+--------+--
|
|
* / \
|
|
* 8 12
|
|
* +----+------------+
|
|
* | ecn| dscp |
|
|
* +----+------------+
|
|
*/
|
|
TEST_ASSERT_EQUAL_INT(((TEST_UINT8 & 0x03) << 4) | ((OTHER_BYTE & 0xf0) >> 4),
|
|
ipv6_hdr_get_tc_dscp((ipv6_hdr_t *)val));
|
|
}
|
|
|
|
static void test_ipv6_hdr_set_fl(void)
|
|
{
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { 0, TEST_UINT8, 0, 0 };
|
|
|
|
ipv6_hdr_set_fl((ipv6_hdr_t *)val, TEST_UINT32);
|
|
|
|
/*
|
|
* Header format:
|
|
* 8 16 24 32
|
|
* +----+--------+--------------------+
|
|
* | 6 | tc | flow label |
|
|
* +----+--------+--------------------+
|
|
*/
|
|
TEST_ASSERT_EQUAL_INT(0, val[0]);
|
|
TEST_ASSERT_EQUAL_INT((TEST_UINT8 & 0xf0) | ((TEST_UINT32 & 0x000f0000) >> 16),
|
|
val[1]);
|
|
TEST_ASSERT_EQUAL_INT((TEST_UINT32 & 0x0000ff00) >> 8,
|
|
val[2]);
|
|
TEST_ASSERT_EQUAL_INT((TEST_UINT32 & 0x000000ff), val[3]);
|
|
}
|
|
|
|
static void test_ipv6_hdr_get_fl(void)
|
|
{
|
|
uint8_t val[sizeof(ipv6_hdr_t)] = { TEST_UINT8, OTHER_BYTE, 0, 0 };
|
|
|
|
/*
|
|
* Header format:
|
|
* 8 16 24 32
|
|
* +----+--------+--------------------+
|
|
* | 6 | tc | flow label |
|
|
* +----+--------+--------------------+
|
|
*/
|
|
TEST_ASSERT_EQUAL_INT((uint32_t)(OTHER_BYTE & 0x0f) << 16,
|
|
ipv6_hdr_get_fl((ipv6_hdr_t *)val));
|
|
}
|
|
|
|
static void test_ipv6_hdr_inet_csum__initial_sum_overflows(void)
|
|
{
|
|
uint16_t sum = 0xffff;
|
|
uint16_t res = 0, payload_len = 0;
|
|
uint8_t val[] = {
|
|
0x60, 0x00, 0x00, 0x00, 0x00, 0x0c, 0x3a, 0x40, /* IPv6 header */
|
|
0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0xc8, 0x86, 0xcd, 0xff, 0xfe, 0x0f, 0xce, 0x49,
|
|
0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x18, 0xaa, 0x2d, 0xff, 0xfe, 0x44, 0x43, 0xac
|
|
};
|
|
|
|
/* calculate checksum of pseudo header */
|
|
res = ipv6_hdr_inet_csum(sum, (ipv6_hdr_t *)&val, PROTNUM_ICMPV6,
|
|
payload_len);
|
|
res = ~res; /* take 1's-complement for correct checksum */
|
|
|
|
TEST_ASSERT_EQUAL_INT(0x1749, res);
|
|
}
|
|
|
|
static void test_ipv6_hdr_inet_csum__initial_sum_0(void)
|
|
{
|
|
/* source: https://www.cloudshark.org/captures/ea72fbab241b (No. 56) */
|
|
uint16_t res = 0, payload_len;
|
|
uint8_t val[] = {
|
|
0x60, 0x00, 0x00, 0x00, 0x00, 0x38, 0x3a, 0xff, /* IPv6 header */
|
|
0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x5a, 0x6d, 0x8f, 0xff, 0xfe, 0x56, 0x30, 0x09,
|
|
0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
|
|
0x86, 0x00, 0x00, 0x00, 0x40, 0x58, 0x07, 0x08, /* ICMPv6 payload */
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* set checksum to 0 */
|
|
0x03, 0x04, 0x40, 0xc0, 0x00, 0x00, 0x00, 0x1e,
|
|
0x00, 0x00, 0x00, 0x14, 0x00, 0x00, 0x00, 0x00,
|
|
0x20, 0x02, 0x18, 0x3d, 0xdb, 0xa4, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x01, 0x01, 0x58, 0x6d, 0x8f, 0x56, 0x30, 0x09
|
|
};
|
|
|
|
payload_len = sizeof(val) - sizeof(ipv6_hdr_t);
|
|
|
|
/* calculate checksum of pseudo header */
|
|
res = ipv6_hdr_inet_csum(0, (ipv6_hdr_t *)&val, PROTNUM_ICMPV6,
|
|
payload_len);
|
|
/* calculate checksum of payload */
|
|
res = inet_csum(res, val + sizeof(ipv6_hdr_t), payload_len);
|
|
res = ~res; /* take 1's-complement for correct checksum */
|
|
|
|
TEST_ASSERT_EQUAL_INT(0xab32, res);
|
|
}
|
|
|
|
static void test_ipv6_ext_frag_get(void)
|
|
{
|
|
static const uint8_t fix_data1[] = {
|
|
PROTNUM_SNP, 0x00, 0x12, 0x29, 0xAB, 0x4A, 0x2D, 0xF3
|
|
};
|
|
static const uint8_t fix_data2[] = {
|
|
PROTNUM_WB_EXPAK, 0x00, 0x1D, 0x78, 0xAE, 0x49, 0x01, 0x9F
|
|
};
|
|
const ipv6_ext_frag_t *frag_hdr;
|
|
|
|
frag_hdr = (ipv6_ext_frag_t *)fix_data1;
|
|
TEST_ASSERT_EQUAL_INT(PROTNUM_SNP, frag_hdr->nh);
|
|
TEST_ASSERT_EQUAL_INT(0U, frag_hdr->resv);
|
|
TEST_ASSERT_EQUAL_INT(4648, ipv6_ext_frag_get_offset(frag_hdr));
|
|
TEST_ASSERT(ipv6_ext_frag_more(frag_hdr));
|
|
TEST_ASSERT_EQUAL_INT(0xAB4A2DF3, byteorder_ntohl(frag_hdr->id));
|
|
|
|
frag_hdr = (ipv6_ext_frag_t *)fix_data2;
|
|
TEST_ASSERT_EQUAL_INT(PROTNUM_WB_EXPAK, frag_hdr->nh);
|
|
TEST_ASSERT_EQUAL_INT(0U, frag_hdr->resv);
|
|
TEST_ASSERT_EQUAL_INT(7544, ipv6_ext_frag_get_offset(frag_hdr));
|
|
TEST_ASSERT(!ipv6_ext_frag_more(frag_hdr));
|
|
TEST_ASSERT_EQUAL_INT(0xAE49019F, byteorder_ntohl(frag_hdr->id));
|
|
}
|
|
|
|
static void test_ipv6_ext_frag_set(void)
|
|
{
|
|
ipv6_ext_frag_t frag_hdr = { .nh = PROTNUM_TPPLUSPLUS,
|
|
.id = { .u8 = { 0xA7, 0xE8, 0x3D, 0x35 } } };
|
|
|
|
ipv6_ext_frag_set_offset(&frag_hdr, 0);
|
|
TEST_ASSERT_EQUAL_INT(PROTNUM_TPPLUSPLUS, frag_hdr.nh);
|
|
TEST_ASSERT_EQUAL_INT(0U, frag_hdr.resv);
|
|
TEST_ASSERT_EQUAL_INT(0, ipv6_ext_frag_get_offset(&frag_hdr));
|
|
TEST_ASSERT(!ipv6_ext_frag_more(&frag_hdr));
|
|
TEST_ASSERT_EQUAL_INT(0xA7E83D35, byteorder_ntohl(frag_hdr.id));
|
|
ipv6_ext_frag_set_more(&frag_hdr);
|
|
TEST_ASSERT_EQUAL_INT(PROTNUM_TPPLUSPLUS, frag_hdr.nh);
|
|
TEST_ASSERT_EQUAL_INT(0U, frag_hdr.resv);
|
|
TEST_ASSERT_EQUAL_INT(0, ipv6_ext_frag_get_offset(&frag_hdr));
|
|
TEST_ASSERT(ipv6_ext_frag_more(&frag_hdr));
|
|
TEST_ASSERT_EQUAL_INT(0xA7E83D35, byteorder_ntohl(frag_hdr.id));
|
|
|
|
ipv6_ext_frag_set_offset(&frag_hdr, 504);
|
|
TEST_ASSERT_EQUAL_INT(PROTNUM_TPPLUSPLUS, frag_hdr.nh);
|
|
TEST_ASSERT_EQUAL_INT(0U, frag_hdr.resv);
|
|
TEST_ASSERT_EQUAL_INT(504, ipv6_ext_frag_get_offset(&frag_hdr));
|
|
TEST_ASSERT(!ipv6_ext_frag_more(&frag_hdr));
|
|
TEST_ASSERT_EQUAL_INT(0xA7E83D35, byteorder_ntohl(frag_hdr.id));
|
|
}
|
|
|
|
static Test *tests_ipv6_hdr_tests(void)
|
|
{
|
|
EMB_UNIT_TESTFIXTURES(fixtures) {
|
|
new_TestFixture(test_ipv6_hdr_set_version),
|
|
new_TestFixture(test_ipv6_hdr_get_version),
|
|
new_TestFixture(test_ipv6_hdr_is_ipv6_hdr__false),
|
|
new_TestFixture(test_ipv6_hdr_is_ipv6_hdr__true),
|
|
new_TestFixture(test_ipv6_hdr_set_tc),
|
|
new_TestFixture(test_ipv6_hdr_set_tc_ecn),
|
|
new_TestFixture(test_ipv6_hdr_set_tc_dscp),
|
|
new_TestFixture(test_ipv6_hdr_get_tc),
|
|
new_TestFixture(test_ipv6_hdr_get_tc_ecn),
|
|
new_TestFixture(test_ipv6_hdr_get_tc_dscp),
|
|
new_TestFixture(test_ipv6_hdr_set_fl),
|
|
new_TestFixture(test_ipv6_hdr_get_fl),
|
|
new_TestFixture(test_ipv6_hdr_inet_csum__initial_sum_overflows),
|
|
new_TestFixture(test_ipv6_hdr_inet_csum__initial_sum_0),
|
|
};
|
|
|
|
EMB_UNIT_TESTCALLER(ipv6_hdr_tests, NULL, NULL, fixtures);
|
|
|
|
return (Test *)&ipv6_hdr_tests;
|
|
}
|
|
|
|
static Test *tests_ipv6_ext_frag_tests(void)
|
|
{
|
|
EMB_UNIT_TESTFIXTURES(fixtures) {
|
|
new_TestFixture(test_ipv6_ext_frag_get),
|
|
new_TestFixture(test_ipv6_ext_frag_set),
|
|
};
|
|
|
|
EMB_UNIT_TESTCALLER(ipv6_ext_frag_tests, NULL, NULL, fixtures);
|
|
|
|
return (Test *)&ipv6_ext_frag_tests;
|
|
}
|
|
|
|
void tests_ipv6_hdr(void)
|
|
{
|
|
TESTS_RUN(tests_ipv6_hdr_tests());
|
|
TESTS_RUN(tests_ipv6_ext_frag_tests());
|
|
}
|
|
/** @} */
|