/* * 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. */ /** * @defgroup net_dhcpv6_client DHCPv6 client * @ingroup net_dhcpv6 * @brief DHCPv6 client implementation * @{ * * @file * @brief DHCPv6 client definitions * * @author Martine Lenders */ #ifndef NET_DHCPV6_CLIENT_H #define NET_DHCPV6_CLIENT_H #include "byteorder.h" #include "event.h" #include "net/ipv6/addr.h" #include "thread.h" #ifdef __cplusplus extern "C" { #endif /** * @name Auto-initialization parameters * @{ */ #ifndef DHCPV6_CLIENT_STACK_SIZE #define DHCPV6_CLIENT_STACK_SIZE (THREAD_STACKSIZE_DEFAULT) /**< stack size */ #endif #ifndef DHCPV6_CLIENT_PRIORITY #define DHCPV6_CLIENT_PRIORITY (THREAD_PRIORITY_MAIN - 2) /**< priority */ #endif /** @} */ /** * @brief Static length of the DUID */ #define DHCPV6_CLIENT_DUID_LEN (sizeof(dhcpv6_duid_l2_t) + 8U) #define DHCPV6_CLIENT_BUFLEN (256) /**< default length for send and receive buffer */ /** * @defgroup net_dhcpv6_conf DHCPv6 compile configurations * @ingroup config * @{ */ /** * @brief Maximum number of prefix leases to be stored * * Should be equal to the number of downstream interfaces * that are to be configured via DHCPv6 */ #ifndef CONFIG_DHCPV6_CLIENT_PFX_LEASE_MAX #define CONFIG_DHCPV6_CLIENT_PFX_LEASE_MAX (1U) #endif /** * @brief Maximum number of address leases to be stored */ #ifndef CONFIG_DHCPV6_CLIENT_ADDR_LEASE_MAX #define CONFIG_DHCPV6_CLIENT_ADDR_LEASE_MAX (1U) #endif /** * @brief Number of addresses needed for using DHCPv6 IA_NA. * * @note Used for calculation of @ref CONFIG_GNRC_NETIF_IPV6_ADDRS_NUMOF. * Set to 0 if `dhcpv6_client_ia_na` is not included. */ #if defined(MODULE_DHCPV6_CLIENT_IA_NA) || defined(DOXYGEN) #define DHCPV6_CLIENT_ADDRS_NUMOF ((int)(CONFIG_DHCPV6_CLIENT_ADDR_LEASE_MAX)) #else #define DHCPV6_CLIENT_ADDRS_NUMOF (0) #endif /** * @brief MUD URL (must use the https:// scheme) * For more info, see the [definitions](@ref net_dhcpv6_mud_url_option) below */ #ifndef CONFIG_DHCPV6_CLIENT_MUD_URL #define CONFIG_DHCPV6_CLIENT_MUD_URL "https://example.org" #endif /** @} */ /** * @name DHCPv6 unique identifier (DUID) definitions * @see [RFC 8415, section 11](https://tools.ietf.org/html/rfc8415#section-11) * @{ */ /** * @brief DUID based on link-layer address plus time */ typedef struct __attribute__((packed)) { network_uint16_t type; /**< @ref DHCPV6_DUID_TYPE_L2 */ network_uint16_t l2type; /**< [hardware type](@ref net_arp_hwtype)) */ /* link-layer address follows this header */ } dhcpv6_duid_l2_t; #if defined(MODULE_AUTO_INIT_DHCPV6_CLIENT) || defined(DOXYGEN) /** * @brief Auto-initializes the client in its own thread * * @note Only available with (and called by) the `dhcpv6_client_auto_init` * module. */ void dhcpv6_client_auto_init(void); #endif /* MODULE_DHCPV6_CLIENT_AUTO_INIT */ /** * @brief Initializes the client * * @pre `event_queue->waiter != NULL` * * @param[in] event_queue Event queue to use with the client. Needs to be * initialized in the handler thread. * @param[in] netif The network interface the client should listen on. * SOCK_ADDR_ANY_NETIF for any interface */ void dhcpv6_client_init(event_queue_t *event_queue, uint16_t netif); /** * @brief Let the server start listening * * @pre @ref dhcpv6_client_init() was called (i.e. the internal event queue of * he client was set). * * This needs to be called *after* all desired [configuration functions] * (@ref net_dhcpv6_client_conf) and @ref dhcpv6_client_init() were called. */ void dhcpv6_client_start(void); /** * @name Configuration functions * @anchor net_dhcpv6_client_conf * @{ */ /** * @brief Configures the client to request prefix delegation for a network * interface from a server * * @pre Module `dhcpv6_client_ia_pd` is compiled in. * @pre `pfx_len <= 128` * * Without module `dhcpv6_client_ia_pd` and `NDEBUG` set this function is a NOP. * Without module `dhcpv6_client_ia_pd` and `NDEBUG` unset this function will * abort the running code on a failed assertion. * * @param[in] netif The interface to request the prefix delegation for. * @param[in] pfx_len The desired length of the prefix (note that the server * might not consider this request). Must be <= 128 * * @retval 0 on success * @retval -ENOMEM when there is no lease entry available anymore * @retval -ENOTSUP when module `dhcpv6_client_ia_pd` is not being used */ int dhcpv6_client_req_ia_pd(unsigned netif, unsigned pfx_len); /** @} */ /** * @brief Configures the client to request non-temporary addresses for a network * interface from a server * @note For multi-hop WPAN meshes a DHCPv6 relay (which is not implemented in * RIOT yet) is required, as DHCPv6 only acts in link scope. * * @param[in] netif The interface to request non-temporaty addresses for. * * @retval 0 on success * @retval -ENOMEM when there is no lease entry available anymore * @retval -ENOTSUP when module `dhcpv6_client_ia_na` is not being used */ int dhcpv6_client_req_ia_na(unsigned netif); /** @} */ /** * @name Stack-specific functions * * These functions need to be provided by the network-stack implementation. * @{ */ /** * @brief Get the link-layer address DUID for the client * * @param[in] netif The network interface the client is bound to. May be * SOCK_ADDR_ANY_NETIF for any interface. * @param[out] duid The resulting DUID. * * @return length of the @p duid on success. * @return 0, on error. */ unsigned dhcpv6_client_get_duid_l2(unsigned netif, dhcpv6_duid_l2_t *duid); /** * @brief Configures a prefix delegation lease that is provided by the server. * * @param[in] netif Network interface the prefix delegation was for. * @param[in] pfx Prefix for the prefix delegation. * @param[in] pfx_len Length of @p pfx in bits. * @param[in] valid Valid lifetime of the prefix delegation. * @param[in] pref Preferred lifetime of the prefix delegation. */ void dhcpv6_client_conf_prefix(unsigned netif, const ipv6_addr_t *pfx, unsigned pfx_len, uint32_t valid, uint32_t pref); /** * @brief Configures a address lease that is provided by the server. * * @param[in] netif Network interface the address was for. * @param[in] addr The assigned address. * @param[in] valid Valid lifetime of the address. * @param[in] pref Preferred lifetime of the address. */ static inline void dhcpv6_client_conf_addr(unsigned netif, const ipv6_addr_t *addr, uint32_t valid, uint32_t pref) { dhcpv6_client_conf_prefix(netif, addr, IPV6_ADDR_BIT_LEN, valid, pref); } /** * @brief Checks if the given network interface is configured * to use DHCPv6 IA_NA * * @param[in] netif Network interface to check. * * @return true, if the network interface is set up for IA_NA. */ bool dhcpv6_client_check_ia_na(unsigned netif); /** * @brief Determines how long the prefix delegation lease is still valid. * * @param[in] netif Network interface the prefix delegation was for. * @param[in] pfx Prefix of the prefix delegation * @param[in] pfx_len Length of @p pfx in bits. * * @return Remaining valid lifetime of the prefix delegation lease in seconds. */ uint32_t dhcpv6_client_prefix_valid_until(unsigned netif, const ipv6_addr_t *pfx, unsigned pfx_len); /** * @brief Determines how long the address lease is still valid. * * @param[in] netif Network interface the address was for. * @param[in] addr The assigned address. * * @return Remaining valid lifetime of the address lease in seconds. */ static inline uint32_t dhcpv6_client_addr_valid_until(unsigned netif, const ipv6_addr_t *addr) { return dhcpv6_client_prefix_valid_until(netif, addr, IPV6_ADDR_BIT_LEN); } /** @} */ /** * @name DHCPv6 Manufacturer Usage Description (MUD) URL option definitions * @see [RFC 8520, section 10](https://tools.ietf.org/html/rfc8520#section-10) * @anchor net_dhcpv6_mud_url_option * @{ */ /** * @brief Length for the send buffer if a MUD URL is included in the DHCP client's packets * * @note Only (re)defined by the `dhcpv6_client_mud_url` pseudo-module. */ #if defined(MODULE_DHCPV6_CLIENT_MUD_URL) || defined(DOXYGEN) #define DHCPV6_CLIENT_SEND_BUFLEN (DHCPV6_CLIENT_BUFLEN + 256) #else #define DHCPV6_CLIENT_SEND_BUFLEN (DHCPV6_CLIENT_BUFLEN) #endif /** * @brief Maximal length of a MUD URL */ #define MAX_MUD_URL_LENGTH (0xFF - sizeof(dhcpv6_opt_mud_url_t)) /** * @brief Definition of DHCPv6 client configuration modes. */ enum { DHCPV6_CLIENT_CONF_MODE_INACTIVE, DHCPV6_CLIENT_CONF_MODE_STATEFUL, DHCPV6_CLIENT_CONF_MODE_STATELESS, }; /** * @brief Changes the DHCPv6 client's configuration mode. * * @param[in] configuration_mode The new configuration mode. */ void dhcpv6_client_set_conf_mode(uint8_t configuration_mode); /** * @brief Retrieves the DHCPv6 client's current configuration mode. * * @return The current configuration mode. */ uint8_t dhcpv6_client_get_conf_mode(void); /** @} */ #ifdef __cplusplus } #endif #endif /* NET_DHCPV6_CLIENT_H */ /** @} */