1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 01:32:44 +01:00
RIOT/tests/unittests/tests-ipv6_hdr/tests-ipv6_hdr.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());
}
/** @} */