1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/tests/unittests/tests-gnrc_ipv6_nib/tests-gnrc_ipv6_nib-ft.c
2022-10-27 16:31:59 +02:00

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;
}