mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-15 10:32:44 +01:00
761 lines
31 KiB
C
761 lines
31 KiB
C
/*
|
|
* Copyright (C) 2016 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 <m.lenders@fu-berlin.de>
|
|
*/
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include "bitfield.h"
|
|
#include "net/ipv6/addr.h"
|
|
#include "net/gnrc/ipv6/nib.h"
|
|
#include "net/gnrc/ipv6/nib/ft.h"
|
|
|
|
#include "_nib-internal.h"
|
|
|
|
#include "unittests-constants.h"
|
|
|
|
#include "tests-gnrc_ipv6_nib.h"
|
|
|
|
#define LINK_LOCAL_PREFIX { 0xfe, 0x08, 0, 0, 0, 0, 0, 0 }
|
|
#define GLOBAL_PREFIX { 0x20, 0x01, 0x0d, 0xb8, 0, 0, 0, 0 }
|
|
#define L2ADDR { 0x90, 0xd5, 0x8e, 0x8c, 0x92, 0x43, 0x73, 0x5c }
|
|
#define GLOBAL_PREFIX_LEN (30)
|
|
#define IFACE (6)
|
|
|
|
static void set_up(void)
|
|
{
|
|
evtimer_event_t *tmp;
|
|
|
|
for (evtimer_event_t *ptr = _nib_evtimer.events;
|
|
(ptr != NULL) && (tmp = (ptr->next), 1);
|
|
ptr = tmp) {
|
|
evtimer_del((evtimer_t *)(&_nib_evtimer), ptr);
|
|
}
|
|
_nib_init();
|
|
}
|
|
|
|
/*
|
|
* Tries to get a route from an empty forwarding table.
|
|
* Expected result: gnrc_ipv6_nib_ft_get() returns -ENETUNREACH
|
|
*/
|
|
static void test_nib_ft_get__ENETUNREACH_empty(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
|
|
TEST_ASSERT_EQUAL_INT(-ENETUNREACH, gnrc_ipv6_nib_ft_get(&dst, NULL, &fte));
|
|
}
|
|
|
|
/*
|
|
* Adds a route (not the default route) to the forwarding table, then tries to
|
|
* get another route.
|
|
* Expected result: gnrc_ipv6_nib_ft_get() returns -ENETUNREACH
|
|
*/
|
|
static void test_nib_ft_get__ENETUNREACH_no_def_route(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
static const ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
dst.u16[0].u16--;
|
|
TEST_ASSERT_EQUAL_INT(-ENETUNREACH, gnrc_ipv6_nib_ft_get(&dst, NULL, &fte));
|
|
}
|
|
|
|
/*
|
|
* Adds the default route to the forwarding table, then tries to get an
|
|
* arbitrary route.
|
|
* Expected result: gnrc_ipv6_nib_ft_get() returns the configured default route
|
|
*/
|
|
static void test_nib_ft_get__success1(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
static const ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(NULL, 0, &next_hop, IFACE, 0));
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_get(&dst, NULL, &fte));
|
|
TEST_ASSERT(ipv6_addr_is_unspecified(&fte.dst));
|
|
TEST_ASSERT(ipv6_addr_equal(&next_hop, &fte.next_hop));
|
|
TEST_ASSERT_EQUAL_INT(0, fte.dst_len);
|
|
/* we can't make any sure assumption on fte.primary */
|
|
TEST_ASSERT_EQUAL_INT(IFACE, fte.iface);
|
|
}
|
|
|
|
/*
|
|
* Adds an arbitrary route to the forwarding table, then tries to get an address
|
|
* with the same prefix.
|
|
* Expected result: gnrc_ipv6_nib_ft_get() returns the configured route
|
|
*/
|
|
static void test_nib_ft_get__success2(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
static const ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_get(&dst, NULL, &fte));
|
|
TEST_ASSERT(ipv6_addr_match_prefix(&dst, &fte.dst) >= GLOBAL_PREFIX_LEN);
|
|
TEST_ASSERT(ipv6_addr_equal(&next_hop, &fte.next_hop));
|
|
TEST_ASSERT_EQUAL_INT(GLOBAL_PREFIX_LEN, fte.dst_len);
|
|
/* we can't make any sure assumption on fte.primary */
|
|
TEST_ASSERT_EQUAL_INT(IFACE, fte.iface);
|
|
}
|
|
|
|
/*
|
|
* Adds two routes to the forwarding table that differ in their last byte
|
|
* (prefixes have same length), then tries to get an address with the same
|
|
* prefix as the first route.
|
|
* Expected result: gnrc_ipv6_nib_ft_get() returns the first configured route
|
|
*/
|
|
static void test_nib_ft_get__success3(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
static const ipv6_addr_t dst1 = { .u64 = { { .u8 = GLOBAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
static const ipv6_addr_t next_hop1 = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
static const ipv6_addr_t next_hop2 = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 + 1 } } };
|
|
ipv6_addr_t dst2 = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
|
|
bf_toggle(dst2.u8, GLOBAL_PREFIX_LEN);
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst1, GLOBAL_PREFIX_LEN,
|
|
&next_hop1, IFACE, 0));
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst2, GLOBAL_PREFIX_LEN,
|
|
&next_hop2, IFACE, 0));
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_get(&dst1, NULL, &fte));
|
|
TEST_ASSERT(ipv6_addr_match_prefix(&dst1, &fte.dst) >= GLOBAL_PREFIX_LEN);
|
|
TEST_ASSERT(ipv6_addr_equal(&next_hop1, &fte.next_hop));
|
|
TEST_ASSERT_EQUAL_INT(GLOBAL_PREFIX_LEN, fte.dst_len);
|
|
/* we can't make any sure assumption on fte.primary */
|
|
TEST_ASSERT_EQUAL_INT(IFACE, fte.iface);
|
|
}
|
|
|
|
/*
|
|
* Adds two routes to the forwarding table that only differ in their prefix
|
|
* length by one bit me length, then tries to get an address with the same
|
|
* prefix as the route with the longer prefix.
|
|
* Expected result: gnrc_ipv6_nib_ft_get() returns route with the longer prefix
|
|
*/
|
|
static void test_nib_ft_get__success4(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
static const ipv6_addr_t next_hop1 = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
static const ipv6_addr_t next_hop2 = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 + 1 } } };
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop1, IFACE, 0));
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN - 1,
|
|
&next_hop2, IFACE, 0));
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_get(&dst, NULL, &fte));
|
|
TEST_ASSERT(ipv6_addr_match_prefix(&dst, &fte.dst) >= GLOBAL_PREFIX_LEN);
|
|
TEST_ASSERT(ipv6_addr_equal(&next_hop1, &fte.next_hop));
|
|
TEST_ASSERT_EQUAL_INT(GLOBAL_PREFIX_LEN, fte.dst_len);
|
|
/* we can't make any sure assumption on fte.primary */
|
|
TEST_ASSERT_EQUAL_INT(IFACE, fte.iface);
|
|
}
|
|
|
|
/*
|
|
* Tries to create a forwarding table entry for the default route (::) with
|
|
* NULL as next hop.
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -EINVAL
|
|
*/
|
|
static void test_nib_ft_add__EINVAL_def_route_next_hop_NULL(void)
|
|
{
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
|
|
TEST_ASSERT_EQUAL_INT(-EINVAL, gnrc_ipv6_nib_ft_add(&ipv6_addr_unspecified,
|
|
GLOBAL_PREFIX_LEN,
|
|
NULL, IFACE, 0));
|
|
TEST_ASSERT_EQUAL_INT(-EINVAL, gnrc_ipv6_nib_ft_add(NULL, GLOBAL_PREFIX_LEN,
|
|
NULL, IFACE, 0));
|
|
TEST_ASSERT_EQUAL_INT(-EINVAL, gnrc_ipv6_nib_ft_add(&dst, 0, NULL, IFACE, 0));
|
|
TEST_ASSERT_EQUAL_INT(-EINVAL, gnrc_ipv6_nib_ft_add(NULL, 0, NULL, IFACE, 0));
|
|
}
|
|
|
|
/*
|
|
* Tries to create a route via interface 0.
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -EINVAL
|
|
*/
|
|
static void test_nib_ft_add__EINVAL_iface0(void)
|
|
{
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
static const ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
|
|
TEST_ASSERT_EQUAL_INT(-EINVAL, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, 0, 0));
|
|
}
|
|
|
|
#if CONFIG_GNRC_IPV6_NIB_NUMOF < CONFIG_GNRC_IPV6_NIB_OFFL_NUMOF
|
|
#define MAX_NUMOF (CONFIG_GNRC_IPV6_NIB_NUMOF)
|
|
#else /* CONFIG_GNRC_IPV6_NIB_NUMOF < CONFIG_GNRC_IPV6_NIB_OFFL_NUMOF */
|
|
#define MAX_NUMOF (CONFIG_GNRC_IPV6_NIB_OFFL_NUMOF)
|
|
#endif
|
|
|
|
/*
|
|
* Creates CONFIG_GNRC_IPV6_NIB_DEFAULT_ROUTER_NUMOF default route entries and then
|
|
* tries to create another one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_def_router(void)
|
|
{
|
|
ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
|
|
for (unsigned i = 0; i < CONFIG_GNRC_IPV6_NIB_DEFAULT_ROUTER_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(NULL, 0, &next_hop,
|
|
IFACE, 0));
|
|
next_hop.u64[1].u64++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(NULL, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates CONFIG_GNRC_IPV6_NIB_OFFL_NUMOF routes with different destinations of same
|
|
* prefix lengths to the same next hop and then tries to create another one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_dst(void)
|
|
{
|
|
ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
|
|
for (unsigned i = 0; i < CONFIG_GNRC_IPV6_NIB_OFFL_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
NULL, IFACE, 0));
|
|
dst.u16[0].u16--;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
NULL, IFACE, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates CONFIG_GNRC_IPV6_NIB_OFFL_NUMOF routes with destinations of different
|
|
* prefix lengths to the same next hop and then tries to create another one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_dst_len(void)
|
|
{
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
unsigned dst_len = GLOBAL_PREFIX_LEN;
|
|
|
|
for (unsigned i = 0; i < CONFIG_GNRC_IPV6_NIB_OFFL_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, dst_len,
|
|
NULL, IFACE, 0));
|
|
dst_len--;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, dst_len,
|
|
NULL, IFACE, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates CONFIG_GNRC_IPV6_NIB_OFFL_NUMOF routes with different destination of
|
|
* different prefix lengths to the same next hop and then tries to create
|
|
* another one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_dst_dst_len(void)
|
|
{
|
|
ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
unsigned dst_len = GLOBAL_PREFIX_LEN;
|
|
|
|
for (unsigned i = 0; i < CONFIG_GNRC_IPV6_NIB_OFFL_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, dst_len,
|
|
NULL, IFACE, 0));
|
|
dst.u16[0].u16--;
|
|
dst_len--;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, dst_len,
|
|
NULL, IFACE, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates MAX_NUMOF routes with the same destination different next hops and
|
|
* then tries to create another one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_next_hop(void)
|
|
{
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
|
|
for (unsigned i = 0; i < MAX_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
next_hop.u64[1].u64++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates MAX_NUMOF routes with different destinations of same prefix lengths
|
|
* to the different next hops and then tries to create another one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_dst_next_hop(void)
|
|
{
|
|
ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
|
|
for (unsigned i = 0; i < MAX_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
dst.u16[0].u16--;
|
|
next_hop.u64[1].u64++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates MAX_NUMOF routes with different destinations of same prefix lengths
|
|
* to different next hops and then tries to create another one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_dst_dst_len_next_hop(void)
|
|
{
|
|
ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
unsigned dst_len = GLOBAL_PREFIX_LEN;
|
|
|
|
for (unsigned i = 0; i < MAX_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, dst_len, &next_hop,
|
|
IFACE, 0));
|
|
dst.u16[0].u16--;
|
|
dst_len--;
|
|
next_hop.u64[1].u64++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, dst_len,
|
|
&next_hop, IFACE, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates MAX_NUMOF routes with different destinations of same prefix lengths
|
|
* to the same next hop but on different interfaces and then tries to create
|
|
* another one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_dst_iface(void)
|
|
{
|
|
ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
unsigned iface = IFACE;
|
|
|
|
for (unsigned i = 0; i < MAX_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
NULL, iface, 0));
|
|
dst.u16[0].u16--;
|
|
iface++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
NULL, iface, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates MAX_NUMOF routes with destinations of different prefix lengths to the
|
|
* same next hop but on different interfaces and then tries to create another
|
|
* one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_dst_len_iface(void)
|
|
{
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
unsigned dst_len = GLOBAL_PREFIX_LEN;
|
|
unsigned iface = IFACE;
|
|
|
|
for (unsigned i = 0; i < MAX_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, dst_len,
|
|
NULL, iface, 0));
|
|
dst_len--;
|
|
iface++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, dst_len,
|
|
NULL, iface, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates MAX_NUMOF routes with different destination of different prefix
|
|
* lengths to the same next hop but on different interfaces and then tries to
|
|
* create another one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_dst_dst_len_iface(void)
|
|
{
|
|
ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
unsigned dst_len = GLOBAL_PREFIX_LEN;
|
|
unsigned iface = IFACE;
|
|
|
|
for (unsigned i = 0; i < MAX_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, dst_len,
|
|
NULL, iface, 0));
|
|
dst.u16[0].u16--;
|
|
dst_len--;
|
|
iface++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, dst_len,
|
|
NULL, iface, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates MAX_NUMOF routes with the same destination to different next hops and
|
|
* interfaces and then tries to create another one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_next_hop_iface(void)
|
|
{
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
unsigned iface = IFACE;
|
|
|
|
for (unsigned i = 0; i < MAX_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, iface, 0));
|
|
next_hop.u64[1].u64++;
|
|
iface++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, iface, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates MAX_NUMOF routes with different destinations of same prefix lengths
|
|
* to different next hops and interfaces and then tries to create another one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_dst_next_hop_iface(void)
|
|
{
|
|
ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
unsigned iface = IFACE;
|
|
|
|
for (unsigned i = 0; i < MAX_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, iface, 0));
|
|
dst.u16[0].u16--;
|
|
next_hop.u64[1].u64++;
|
|
iface++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, iface, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates MAX_NUMOF routes with different destinations of same prefix lengths
|
|
* to the different next hops and interfaces and then tries to create another
|
|
* one
|
|
* Expected result: gnrc_ipv6_nib_ft_add() returns -ENOMEM
|
|
*/
|
|
static void test_nib_ft_add__ENOMEM_diff_dst_dst_len_next_hop_iface(void)
|
|
{
|
|
ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
unsigned dst_len = GLOBAL_PREFIX_LEN;
|
|
unsigned iface = IFACE;
|
|
|
|
for (unsigned i = 0; i < MAX_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, dst_len, &next_hop,
|
|
iface, 0));
|
|
dst.u16[0].u16--;
|
|
dst_len--;
|
|
next_hop.u64[1].u64++;
|
|
iface++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gnrc_ipv6_nib_ft_add(&dst, dst_len,
|
|
&next_hop, iface, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates MAX_NUMOF routes with different destinations to different next hops
|
|
* and interfaces and then tries add another equal to the last.
|
|
* Expected result: should return not NULL (the last)
|
|
*/
|
|
static void test_nib_ft_add__success_duplicate(void)
|
|
{
|
|
ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
unsigned dst_len = GLOBAL_PREFIX_LEN;
|
|
unsigned iface = IFACE;
|
|
|
|
for (unsigned i = 0; i < MAX_NUMOF; i++) {
|
|
dst.u16[0].u16--;
|
|
dst_len--;
|
|
next_hop.u64[1].u64++;
|
|
iface++;
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, dst_len, &next_hop,
|
|
iface, 0));
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, dst_len,
|
|
&next_hop, iface, 0));
|
|
}
|
|
|
|
/*
|
|
* Creates a route with no next hop address then adds another with equal prefix
|
|
* and interface to the last, but with a next hop address
|
|
* Expected result: there should only be one route (with the configuration of
|
|
* the second)
|
|
*/
|
|
static void test_nib_ft_add__success_overwrite_unspecified(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
void *iter_state = NULL;
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
static const ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN, NULL,
|
|
IFACE, 0));
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
TEST_ASSERT(gnrc_ipv6_nib_ft_iter(NULL, 0, &iter_state, &fte));
|
|
TEST_ASSERT(ipv6_addr_equal(&fte.next_hop, &next_hop));
|
|
TEST_ASSERT(!gnrc_ipv6_nib_ft_iter(NULL, 0, &iter_state, &fte));
|
|
}
|
|
|
|
/*
|
|
* Creates a route
|
|
* Expected result: a new entry should exist and contain the given prefix,
|
|
* interface, and lifetimes
|
|
*/
|
|
static void test_nib_ft_add__success(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
void *iter_state = NULL;
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
static const ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
TEST_ASSERT(gnrc_ipv6_nib_ft_iter(NULL, 0, &iter_state, &fte));
|
|
TEST_ASSERT(ipv6_addr_match_prefix(&fte.dst, &dst) >= GLOBAL_PREFIX_LEN);
|
|
TEST_ASSERT_EQUAL_INT(GLOBAL_PREFIX_LEN, fte.dst_len);
|
|
TEST_ASSERT(ipv6_addr_equal(&fte.next_hop, &next_hop));
|
|
TEST_ASSERT_EQUAL_INT(IFACE, fte.iface);
|
|
TEST_ASSERT_EQUAL_INT(0, fte.primary);
|
|
TEST_ASSERT(!gnrc_ipv6_nib_ft_iter(NULL, 0, &iter_state, &fte));
|
|
}
|
|
|
|
/*
|
|
* Creates a default route
|
|
* Expected result: a new entry should exist and contain the given prefix,
|
|
* interface, and lifetimes and it should be the primary default route
|
|
*/
|
|
static void test_nib_ft_add__success_dr(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
void *iter_state = NULL;
|
|
static const ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(NULL, 0, &next_hop, IFACE, 0));
|
|
TEST_ASSERT(gnrc_ipv6_nib_ft_iter(NULL, 0, &iter_state, &fte));
|
|
TEST_ASSERT(ipv6_addr_is_unspecified(&fte.dst));
|
|
TEST_ASSERT_EQUAL_INT(0, fte.dst_len);
|
|
TEST_ASSERT(ipv6_addr_equal(&fte.next_hop, &next_hop));
|
|
TEST_ASSERT_EQUAL_INT(IFACE, fte.iface);
|
|
TEST_ASSERT_EQUAL_INT(1, fte.primary);
|
|
TEST_ASSERT(!gnrc_ipv6_nib_ft_iter(NULL, 0, &iter_state, &fte));
|
|
}
|
|
|
|
/*
|
|
* Creates MAX_NUMOF routes with different destinations of to the different
|
|
* next hops and interfaces and then tries to delete one with yet another
|
|
* destination, next hop and interface.
|
|
* Expected result: There should be still MAX_NUMOF entries in the forwarding
|
|
* table
|
|
*/
|
|
static void test_nib_ft_del__unknown(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
void *iter_state = NULL;
|
|
ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
unsigned dst_len = GLOBAL_PREFIX_LEN, iface = IFACE, count = 0;
|
|
|
|
for (unsigned i = 0; i < MAX_NUMOF; i++) {
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, dst_len, &next_hop,
|
|
iface, 0));
|
|
dst.u16[0].u16--;
|
|
dst_len--;
|
|
next_hop.u64[1].u64++;
|
|
iface++;
|
|
}
|
|
gnrc_ipv6_nib_ft_del(&dst, dst_len);
|
|
while(gnrc_ipv6_nib_ft_iter(NULL, 0, &iter_state, &fte)) {
|
|
count++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(MAX_NUMOF, count);
|
|
}
|
|
|
|
/*
|
|
* Creates a route and removes it.
|
|
* Expected result: forwarding table should be empty
|
|
*/
|
|
static void test_nib_ft_del__success(void)
|
|
{
|
|
void *iter_state = NULL;
|
|
static const ipv6_addr_t dst = { .u64 = { { .u8 = GLOBAL_PREFIX } } };
|
|
static const ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&dst, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
gnrc_ipv6_nib_ft_del(&dst, GLOBAL_PREFIX_LEN);
|
|
TEST_ASSERT(!gnrc_ipv6_nib_ft_iter(NULL ,0, &iter_state, &fte));
|
|
}
|
|
|
|
/**
|
|
* Creates three default routes and removes the first one.
|
|
* The prefix list is then iterated.
|
|
* Expected result: there should be two default routes returned, the last
|
|
* two added.
|
|
*/
|
|
static void test_nib_ft_iter__empty_def_route_at_beginning(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
void *iter_state = NULL;
|
|
ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
int count = 0;
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(NULL, 0,
|
|
&next_hop, IFACE, 0));
|
|
next_hop.u16[0].u16++;
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(NULL, 0,
|
|
&next_hop, IFACE, 0));
|
|
next_hop.u16[0].u16++;
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(NULL, 0,
|
|
&next_hop, IFACE, 0));
|
|
/* make first added route the primary default route again */
|
|
next_hop.u16[0].u16 -= 2;
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(NULL, 0,
|
|
&next_hop, IFACE, 0));
|
|
/* remove primary default route */
|
|
gnrc_ipv6_nib_ft_del(NULL, 0);
|
|
next_hop.u16[0].u16++;
|
|
while (gnrc_ipv6_nib_ft_iter(NULL, 0, &iter_state, &fte)) {
|
|
TEST_ASSERT(ipv6_addr_is_unspecified(&fte.dst));
|
|
TEST_ASSERT(ipv6_addr_equal(&fte.next_hop, &next_hop));
|
|
TEST_ASSERT_EQUAL_INT(0, fte.dst_len);
|
|
TEST_ASSERT_EQUAL_INT(IFACE, fte.iface);
|
|
TEST_ASSERT(!fte.primary);
|
|
count++;
|
|
next_hop.u16[0].u16++;
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(2, count);
|
|
}
|
|
|
|
/**
|
|
* Creates three prefix based routes and removes the second one.
|
|
* The prefix list is then iterated.
|
|
* Expected result: there should be two prefix based routes returned, the first
|
|
* and the third one.
|
|
*/
|
|
static void test_nib_ft_iter__empty_pref_route_in_the_middle(void)
|
|
{
|
|
gnrc_ipv6_nib_ft_t fte;
|
|
void *iter_state = NULL;
|
|
ipv6_addr_t route = { .u64 = { { .u8 = GLOBAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
const ipv6_addr_t next_hop = { .u64 = { { .u8 = LINK_LOCAL_PREFIX },
|
|
{ .u64 = TEST_UINT64 } } };
|
|
int count = 0;
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&route, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
route.u16[0].u16++;
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&route, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
route.u16[0].u16++;
|
|
TEST_ASSERT_EQUAL_INT(0, gnrc_ipv6_nib_ft_add(&route, GLOBAL_PREFIX_LEN,
|
|
&next_hop, IFACE, 0));
|
|
route.u16[0].u16--;
|
|
gnrc_ipv6_nib_ft_del(&route, GLOBAL_PREFIX_LEN);
|
|
route.u16[0].u16--;
|
|
while (gnrc_ipv6_nib_ft_iter(NULL, 0, &iter_state, &fte)) {
|
|
TEST_ASSERT(ipv6_addr_match_prefix(&fte.dst, &route) >= GLOBAL_PREFIX_LEN);
|
|
TEST_ASSERT(ipv6_addr_equal(&fte.next_hop, &next_hop));
|
|
TEST_ASSERT_EQUAL_INT(GLOBAL_PREFIX_LEN, fte.dst_len);
|
|
TEST_ASSERT_EQUAL_INT(IFACE, fte.iface);
|
|
count++;
|
|
route.u16[0].u16 += 2; /* we skip the second address */
|
|
}
|
|
TEST_ASSERT_EQUAL_INT(2, count);
|
|
}
|
|
|
|
Test *tests_gnrc_ipv6_nib_ft_tests(void)
|
|
{
|
|
EMB_UNIT_TESTFIXTURES(fixtures) {
|
|
new_TestFixture(test_nib_ft_get__ENETUNREACH_empty),
|
|
new_TestFixture(test_nib_ft_get__ENETUNREACH_no_def_route),
|
|
new_TestFixture(test_nib_ft_get__success1),
|
|
new_TestFixture(test_nib_ft_get__success2),
|
|
new_TestFixture(test_nib_ft_get__success3),
|
|
new_TestFixture(test_nib_ft_get__success4),
|
|
new_TestFixture(test_nib_ft_add__EINVAL_def_route_next_hop_NULL),
|
|
new_TestFixture(test_nib_ft_add__EINVAL_iface0),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_def_router),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_dst),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_dst_len),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_dst_dst_len),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_next_hop),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_dst_next_hop),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_dst_dst_len_next_hop),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_dst_iface),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_dst_len_iface),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_dst_dst_len_iface),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_next_hop_iface),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_dst_next_hop_iface),
|
|
new_TestFixture(test_nib_ft_add__ENOMEM_diff_dst_dst_len_next_hop_iface),
|
|
new_TestFixture(test_nib_ft_add__success_duplicate),
|
|
new_TestFixture(test_nib_ft_add__success_overwrite_unspecified),
|
|
new_TestFixture(test_nib_ft_add__success),
|
|
new_TestFixture(test_nib_ft_add__success_dr),
|
|
new_TestFixture(test_nib_ft_del__unknown),
|
|
new_TestFixture(test_nib_ft_del__success),
|
|
/* most of gnrc_ipv6_nib_ft_iter() is tested during all the tests above */
|
|
new_TestFixture(test_nib_ft_iter__empty_def_route_at_beginning),
|
|
new_TestFixture(test_nib_ft_iter__empty_pref_route_in_the_middle),
|
|
};
|
|
|
|
EMB_UNIT_TESTCALLER(tests, set_up, NULL,
|
|
fixtures);
|
|
|
|
return (Test *)&tests;
|
|
}
|