2018-03-21 12:49:40 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 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>
|
|
|
|
*/
|
|
|
|
|
2020-10-21 15:58:33 +02:00
|
|
|
#include <assert.h>
|
|
|
|
|
2018-03-21 12:49:40 +01:00
|
|
|
#include "log.h"
|
|
|
|
#include "net/arp.h"
|
|
|
|
#include "net/dhcpv6.h"
|
2021-06-05 23:07:38 +02:00
|
|
|
#include "net/gnrc/dhcpv6/client/simple_pd.h"
|
2018-03-21 12:49:40 +01:00
|
|
|
#include "net/gnrc/ipv6/nib/pl.h"
|
2020-02-26 14:55:28 +01:00
|
|
|
#include "net/gnrc/sixlowpan/ctx.h"
|
2018-03-21 12:49:40 +01:00
|
|
|
#include "net/gnrc/netif.h"
|
|
|
|
#include "net/gnrc/rpl.h"
|
|
|
|
#include "net/sock.h"
|
|
|
|
#include "timex.h"
|
2021-07-23 17:40:06 +02:00
|
|
|
#include "evtimer.h"
|
2018-03-21 12:49:40 +01:00
|
|
|
|
|
|
|
#include "net/dhcpv6/client.h"
|
|
|
|
|
2020-10-22 11:35:22 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2018-03-21 12:49:40 +01:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
unsigned dhcpv6_client_get_duid_l2(unsigned iface, dhcpv6_duid_l2_t *duid)
|
|
|
|
{
|
|
|
|
gnrc_netif_t *netif;
|
|
|
|
uint8_t *l2addr = ((uint8_t *)(duid)) + sizeof(dhcpv6_duid_l2_t);
|
|
|
|
int res;
|
|
|
|
|
|
|
|
duid->type = byteorder_htons(DHCPV6_DUID_TYPE_L2);
|
|
|
|
/* TODO make GNRC-independent */
|
|
|
|
if (iface == SOCK_ADDR_ANY_NETIF) {
|
|
|
|
netif = gnrc_netif_iter(NULL);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
netif = gnrc_netif_get_by_pid(iface);
|
|
|
|
}
|
|
|
|
assert(netif != NULL);
|
|
|
|
if ((res = gnrc_netapi_get(netif->pid, NETOPT_ADDRESS_LONG, 0,
|
|
|
|
l2addr, GNRC_NETIF_L2ADDR_MAXLEN)) > 0) {
|
|
|
|
duid->l2type = byteorder_htons(ARP_HWTYPE_EUI64);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
switch (netif->device_type) {
|
2020-04-07 22:01:03 +02:00
|
|
|
case NETDEV_TYPE_SLIP:
|
|
|
|
duid->l2type = byteorder_htons(ARP_HWTYPE_SERIAL);
|
|
|
|
/* L2 address length is 0 */
|
|
|
|
res = 0;
|
|
|
|
break;
|
2018-03-21 12:49:40 +01:00
|
|
|
case NETDEV_TYPE_ETHERNET:
|
|
|
|
case NETDEV_TYPE_BLE:
|
|
|
|
case NETDEV_TYPE_ESP_NOW:
|
|
|
|
if ((res = gnrc_netapi_get(netif->pid,
|
|
|
|
NETOPT_ADDRESS,
|
|
|
|
0, l2addr,
|
|
|
|
GNRC_NETIF_L2ADDR_MAXLEN)) > 0) {
|
|
|
|
duid->l2type = byteorder_htons(ARP_HWTYPE_ETHERNET);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* intentionally falls through */
|
|
|
|
default:
|
|
|
|
LOG_ERROR("DHCPv6 client: Link-layer type of interface %u not supported "
|
|
|
|
"for DUID creation\n", netif->pid);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (uint8_t)res + sizeof(dhcpv6_duid_l2_t);
|
|
|
|
}
|
|
|
|
|
|
|
|
void dhcpv6_client_conf_prefix(unsigned iface, const ipv6_addr_t *pfx,
|
|
|
|
unsigned pfx_len, uint32_t valid,
|
|
|
|
uint32_t pref)
|
|
|
|
{
|
|
|
|
gnrc_netif_t *netif = gnrc_netif_get_by_pid(iface);
|
2021-07-22 17:00:21 +02:00
|
|
|
int idx = gnrc_netif_ipv6_add_prefix(netif, pfx, pfx_len, valid, pref);
|
|
|
|
if (idx >= 0) {
|
|
|
|
gnrc_rpl_configure_root(netif, &netif->ipv6.addrs[idx]);
|
2018-03-21 12:49:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-24 18:03:02 +01:00
|
|
|
bool dhcpv6_client_check_ia_na(unsigned iface)
|
|
|
|
{
|
|
|
|
gnrc_netif_t *netif = gnrc_netif_get_by_pid(iface);
|
|
|
|
|
|
|
|
return netif->ipv6.aac_mode & GNRC_NETIF_AAC_DHCP;
|
|
|
|
}
|
|
|
|
|
|
|
|
int dhcpv6_client_add_addr(unsigned iface, ipv6_addr_t *addr)
|
|
|
|
{
|
|
|
|
gnrc_netif_t *netif = gnrc_netif_get_by_pid(iface);
|
|
|
|
|
|
|
|
DEBUG("DHCPv6 client: ADD IP ADDRESS\n");
|
|
|
|
|
|
|
|
return gnrc_netif_ipv6_addr_add(netif, addr, 64, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void dhcpv6_client_deprecate_addr(unsigned iface, const ipv6_addr_t *addr)
|
|
|
|
{
|
|
|
|
gnrc_netif_t *netif = gnrc_netif_get_by_pid(iface);
|
|
|
|
int i;
|
|
|
|
|
|
|
|
gnrc_netif_acquire(netif);
|
|
|
|
i = gnrc_netif_ipv6_addr_idx(netif, addr);
|
|
|
|
if (i >= 0) {
|
|
|
|
netif->ipv6.addrs_flags[i] &= ~GNRC_NETIF_IPV6_ADDRS_FLAGS_STATE_MASK;
|
|
|
|
netif->ipv6.addrs_flags[i] |= GNRC_NETIF_IPV6_ADDRS_FLAGS_STATE_DEPRECATED;
|
|
|
|
}
|
|
|
|
gnrc_netif_release(netif);
|
|
|
|
}
|
|
|
|
|
|
|
|
void dhcpv6_client_remove_addr(unsigned iface, ipv6_addr_t *addr)
|
|
|
|
{
|
|
|
|
gnrc_netif_t *netif = gnrc_netif_get_by_pid(iface);
|
|
|
|
|
|
|
|
gnrc_netif_ipv6_addr_remove(netif, addr);
|
|
|
|
}
|
|
|
|
|
2018-03-21 12:49:40 +01:00
|
|
|
uint32_t dhcpv6_client_prefix_valid_until(unsigned netif,
|
|
|
|
const ipv6_addr_t *pfx,
|
|
|
|
unsigned pfx_len)
|
|
|
|
{
|
|
|
|
gnrc_ipv6_nib_pl_t ple;
|
|
|
|
void *state = NULL;
|
|
|
|
uint32_t max_valid = 0;
|
2021-07-23 17:40:06 +02:00
|
|
|
uint32_t now = evtimer_now_msec();
|
2018-03-21 12:49:40 +01:00
|
|
|
|
|
|
|
while (gnrc_ipv6_nib_pl_iter(netif, &state, &ple)) {
|
|
|
|
if ((ple.pfx_len == pfx_len) &&
|
2021-07-23 17:40:06 +02:00
|
|
|
(((ple.valid_until - now) / MS_PER_SEC) > max_valid) &&
|
2018-03-21 12:49:40 +01:00
|
|
|
(ipv6_addr_match_prefix(&ple.pfx,
|
|
|
|
pfx) >= ple.pfx_len)) {
|
2021-07-23 17:40:06 +02:00
|
|
|
max_valid = (ple.valid_until - now) / MS_PER_SEC;
|
2018-03-21 12:49:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return max_valid;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @} */
|