mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
155 lines
4.8 KiB
C
155 lines
4.8 KiB
C
/*
|
|
* Copyright (C) 2018-20 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 S. Lenders <m.lenders@fu-berlin.de>
|
|
*/
|
|
|
|
#include "event.h"
|
|
#include "log.h"
|
|
#include "net/dhcpv6/client.h"
|
|
#include "net/ipv6/addr.h"
|
|
#include "net/gnrc.h"
|
|
#include "net/gnrc/ipv6/nib.h"
|
|
#include "net/gnrc/ipv6/nib/ft.h"
|
|
#include "net/gnrc/netif/internal.h"
|
|
|
|
#include "net/gnrc/dhcpv6/client/simple_pd.h"
|
|
|
|
#if IS_USED(MODULE_AUTO_INIT_DHCPV6_CLIENT)
|
|
#error "Module `gnrc_dhcpv6_client_simple_pd` is mutually exclusive to \
|
|
`auto_init_dhcpv6_client`"
|
|
#endif
|
|
|
|
static char _stack[DHCPV6_CLIENT_STACK_SIZE];
|
|
|
|
/**
|
|
* @brief Find upstream network interface
|
|
*
|
|
* Either the one network interface configured at compile-time with @ref
|
|
* CONFIG_GNRC_DHCPV6_CLIENT_6LBR_UPSTREAM is picked or the first network
|
|
* interface that is not a 6LoWPAN interfaces if
|
|
* `CONFIG_GNRC_DHCPV6_CLIENT_6LBR_UPSTREAM` is 0.
|
|
*
|
|
* @return The upstream network interface.
|
|
*/
|
|
static gnrc_netif_t *_find_upstream_netif(void)
|
|
{
|
|
gnrc_netif_t *netif = NULL;
|
|
|
|
if (CONFIG_GNRC_DHCPV6_CLIENT_6LBR_UPSTREAM) {
|
|
return gnrc_netif_get_by_pid(CONFIG_GNRC_DHCPV6_CLIENT_6LBR_UPSTREAM);
|
|
}
|
|
while ((netif = gnrc_netif_iter(netif))) {
|
|
if (!gnrc_netif_is_6lo(netif)) {
|
|
LOG_WARNING("DHCPv6: Selecting interface %d as upstream\n",
|
|
netif->pid);
|
|
return netif;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* @brief Configure upstream netif to be in line with configuration script
|
|
*
|
|
* Set route and link-local address in accordance to
|
|
* `dist/tools/ethos/setup_network.sh`.
|
|
*
|
|
* @note This might not be necessary with a properly set-up DHCPv6 server
|
|
* (automatically configures a route for the delegated prefix) and
|
|
* upstream router (sends periodic router advertisements).
|
|
*
|
|
* @param[in] upstream_netif The upstream netif The upstream netif
|
|
*/
|
|
static void _configure_upstream_netif(gnrc_netif_t *upstream_netif)
|
|
{
|
|
if (IS_ACTIVE(CONFIG_GNRC_DHCPV6_CLIENT_6LBR_STATIC_ROUTE)) {
|
|
ipv6_addr_t addr = {
|
|
.u8 = { 0xfe, 0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1 }
|
|
};
|
|
|
|
/* set default route to host machine (as set-up in setup_network.sh) */
|
|
gnrc_ipv6_nib_ft_add(NULL, 0, &addr, upstream_netif->pid, 0);
|
|
/* set additional link-local address to provide a well-known next hop
|
|
* for static route configuration on the host machine */
|
|
addr.u8[15] = 2;
|
|
gnrc_netif_ipv6_addr_add(upstream_netif, &addr, 64, 0);
|
|
}
|
|
|
|
/* Disable router advertisements on upstream interface. With this, the border
|
|
* router
|
|
* 1. Does not confuse the upstream router to add the border router to its
|
|
* default router list and
|
|
* 2. Solicits upstream Router Advertisements quicker to auto-configure its
|
|
* upstream global address.
|
|
*/
|
|
gnrc_ipv6_nib_change_rtr_adv_iface(upstream_netif, false);
|
|
}
|
|
|
|
/**
|
|
* @brief Configures all 6LoWPAN interfaces to request a 64-bit prefix
|
|
*/
|
|
static void _configure_dhcpv6_client(void)
|
|
{
|
|
gnrc_netif_t *netif = NULL;
|
|
gnrc_netif_t *upstream = _find_upstream_netif();
|
|
while ((netif = gnrc_netif_iter(netif))) {
|
|
if (IS_USED(MODULE_GNRC_DHCPV6_CLIENT_6LBR)
|
|
&& !gnrc_netif_is_6lo(netif)) {
|
|
continue;
|
|
}
|
|
if (netif != upstream) {
|
|
dhcpv6_client_req_ia_pd(netif->pid, 64U);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief The DHCPv6 client thread
|
|
*/
|
|
static void *_dhcpv6_cl_simple_pd_thread(void *args)
|
|
{
|
|
event_queue_t event_queue;
|
|
gnrc_netif_t *upstream_netif = _find_upstream_netif();
|
|
|
|
(void)args;
|
|
if (upstream_netif == NULL) {
|
|
LOG_ERROR("DHCPv6: No upstream interface found!\n");
|
|
return NULL;
|
|
}
|
|
_configure_upstream_netif(upstream_netif);
|
|
/* initialize client event queue */
|
|
event_queue_init(&event_queue);
|
|
/* initialize DHCPv6 client on border interface */
|
|
dhcpv6_client_init(&event_queue, upstream_netif->pid);
|
|
/* configure client to request prefix delegation for WPAN interfaces */
|
|
_configure_dhcpv6_client();
|
|
/* set client configuration mode to stateful */
|
|
dhcpv6_client_set_conf_mode(DHCPV6_CLIENT_CONF_MODE_STATEFUL);
|
|
/* start DHCPv6 client */
|
|
dhcpv6_client_start();
|
|
/* start event loop of DHCPv6 client */
|
|
event_loop(&event_queue); /* never returns */
|
|
return NULL;
|
|
}
|
|
|
|
void gnrc_dhcpv6_client_simple_pd_init(void)
|
|
{
|
|
/* start DHCPv6 client thread to request prefix for WPAN */
|
|
thread_create(_stack, DHCPV6_CLIENT_STACK_SIZE,
|
|
DHCPV6_CLIENT_PRIORITY,
|
|
THREAD_CREATE_STACKTEST,
|
|
_dhcpv6_cl_simple_pd_thread, NULL, "dhcpv6-client");
|
|
}
|
|
|
|
/** @} */
|