mbed-os/features/nanostack/mbed-mesh-api/source/WisunBorderRouter.cpp

501 lines
14 KiB
C++

/*
* Copyright (c) 2020 ARM Limited. All rights reserved.
* SPDX-License-Identifier: Apache-2.0
* Licensed under the Apache License, Version 2.0 (the License); you may
* not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an AS IS BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "WisunInterface.h"
#include "ns_trace.h"
#include "WisunBorderRouter.h"
#include "MeshInterfaceNanostack.h"
#include "net_interface.h"
#include "ip6string.h"
extern "C" {
#include "ws_bbr_api.h"
}
#define TRACE_GROUP "WSBR"
WisunBorderRouter::WisunBorderRouter()
{
// Apply mbed configuration to Wi-SUN BBR
configure();
}
mesh_error_t WisunBorderRouter::start(NetworkInterface *mesh_if, NetworkInterface *backbone_if)
{
if (mesh_if == NULL || backbone_if == NULL) {
return MESH_ERROR_PARAM;
}
WisunInterface *wisun_mesh_if = reinterpret_cast<WisunInterface *>(mesh_if);
return start(wisun_mesh_if, backbone_if);
}
mesh_error_t WisunBorderRouter::start(WisunInterface *mesh_if, NetworkInterface *backbone_if)
{
if (mesh_if == NULL || backbone_if == NULL) {
return MESH_ERROR_PARAM;
}
int8_t mesh_if_id = mesh_if->get_interface_id();
if (mesh_if_id < 0) {
return MESH_ERROR_UNKNOWN;
}
_mesh_if_id = mesh_if_id;
char backbone_if_name[7] = {0};
if (backbone_if->get_interface_name(backbone_if_name) == NULL) {
return MESH_ERROR_UNKNOWN;
}
if (strlen((char *) &backbone_if_name) < 4) {
return MESH_ERROR_UNKNOWN;
}
int backbone_if_id = atoi(&backbone_if_name[3]);
if (backbone_if_id < 0) {
return MESH_ERROR_UNKNOWN;
}
apply_configuration(mesh_if_id);
int ret = ws_bbr_start(mesh_if_id, backbone_if_id);
if (ret < 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::start(NetworkInterface *mesh_if, OnboardNetworkStack::Interface *backbone_if)
{
if (mesh_if == NULL || backbone_if == NULL) {
return MESH_ERROR_PARAM;
}
WisunInterface *wisun_mesh_if = reinterpret_cast<WisunInterface *>(mesh_if);
return start(wisun_mesh_if, backbone_if);
}
mesh_error_t WisunBorderRouter::start(WisunInterface *mesh_if, OnboardNetworkStack::Interface *backbone_if)
{
if (mesh_if == NULL || backbone_if == NULL) {
return MESH_ERROR_PARAM;
}
int8_t mesh_if_id = mesh_if->get_interface_id();
if (mesh_if_id < 0) {
return MESH_ERROR_UNKNOWN;
}
_mesh_if_id = mesh_if_id;
Nanostack::Interface *nano_backbone_if = static_cast<Nanostack::Interface *>(backbone_if);
int8_t backbone_if_id = nano_backbone_if->get_interface_id();
if (backbone_if_id < 0) {
return MESH_ERROR_UNKNOWN;
}
apply_configuration(mesh_if_id);
int ret = ws_bbr_start(mesh_if_id, backbone_if_id);
if (ret < 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
void WisunBorderRouter::stop()
{
if (_mesh_if_id < 0) {
return;
}
ws_bbr_stop(_mesh_if_id);
_mesh_if_id = -1;
}
mesh_error_t WisunBorderRouter::configure()
{
#if defined(MBED_CONF_MBED_MESH_API_RADIUS_SHARED_SECRET) || defined(MBED_CONF_MBED_MESH_API_RADIUS_SERVER_IPV6_ADDRESS)
mesh_error_t status;
#endif
if (_configured) {
// Already configured
return MESH_ERROR_NONE;
}
_configured = true;
#ifdef MBED_CONF_MBED_MESH_API_RADIUS_SHARED_SECRET
const char radius_shared_secret[] = {MBED_CONF_MBED_MESH_API_RADIUS_SHARED_SECRET};
#ifdef MBED_CONF_MBED_MESH_API_RADIUS_SHARED_SECRET_LEN
const uint16_t radius_shared_secret_len = MBED_CONF_MBED_MESH_API_RADIUS_SHARED_SECRET_LEN;
#else
uint16_t radius_shared_secret_len = strlen(radius_shared_secret);
#endif
status = set_radius_shared_secret(radius_shared_secret_len, (uint8_t *) radius_shared_secret);
if (status != MESH_ERROR_NONE) {
tr_error("Failed to set RADIUS shared secret!");
}
#endif
#ifdef MBED_CONF_MBED_MESH_API_RADIUS_SERVER_IPV6_ADDRESS
const char radius_server_ipv6_addr[] = {MBED_CONF_MBED_MESH_API_RADIUS_SERVER_IPV6_ADDRESS};
status = set_radius_server_ipv6_address(radius_server_ipv6_addr);
if (status != MESH_ERROR_NONE) {
tr_error("Failed to set RADIUS server IPv6 address!");
}
#if defined(MBED_CONF_MBED_MESH_API_RADIUS_RETRY_IMIN) || defined(MBED_CONF_MBED_MESH_API_RADIUS_RETRY_IMAX) || defined(MBED_CONF_MBED_MESH_API_RADIUS_RETRY_COUNT)
ws_br_radius_timing_t timing {
.radius_retry_imin = MBED_CONF_MBED_MESH_API_RADIUS_RETRY_IMIN,
.radius_retry_imax = MBED_CONF_MBED_MESH_API_RADIUS_RETRY_IMAX,
.radius_retry_count = MBED_CONF_MBED_MESH_API_RADIUS_RETRY_COUNT
};
status = set_radius_timing(&timing);
if (status != MESH_ERROR_NONE) {
tr_error("Failed to set RADIUS timing parameters!");
}
#endif
#endif
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::apply_configuration(int8_t mesh_if_id)
{
configure();
mesh_error_t status = set_bbr_radius_address();
if (status != MESH_ERROR_NONE) {
tr_error("Failed to apply RADIUS server IPv6 address!");
}
status = set_bbr_radius_shared_secret();
if (status != MESH_ERROR_NONE) {
tr_error("Failed to apply RADIUS server IPv6 address!");
}
status = set_bbr_radius_timing();
if (status != MESH_ERROR_NONE) {
tr_error("Failed to apply RADIUS timing parameters!");
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::set_rpl_parameters(uint8_t dio_interval_min, uint8_t dio_interval_doublings, uint8_t dio_redundancy_constant)
{
int status = ws_bbr_rpl_parameters_set(_mesh_if_id, dio_interval_min, dio_interval_doublings, dio_redundancy_constant);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::get_rpl_parameters(uint8_t *dio_interval_min, uint8_t *dio_interval_doublings, uint8_t *dio_redundancy_constant)
{
int status = ws_bbr_rpl_parameters_get(_mesh_if_id, dio_interval_min, dio_interval_doublings, dio_redundancy_constant);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::validate_rpl_parameters(uint8_t dio_interval_min, uint8_t dio_interval_doublings, uint8_t dio_redundancy_constant)
{
int status = ws_bbr_rpl_parameters_validate(_mesh_if_id, dio_interval_min, dio_interval_doublings, dio_redundancy_constant);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::set_pan_configuration(uint16_t pan_id)
{
int status = ws_bbr_pan_configuration_set(_mesh_if_id, pan_id);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::get_pan_configuration(uint16_t *pan_id)
{
int status = ws_bbr_pan_configuration_get(_mesh_if_id, pan_id);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::validate_pan_configuration(uint16_t pan_id)
{
int status = ws_bbr_pan_configuration_validate(_mesh_if_id, pan_id);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::info_get(ws_br_info_t *info_ptr)
{
bbr_information_t bbr_info = {0};
if (info_ptr == NULL) {
return MESH_ERROR_PARAM;
}
int status = ws_bbr_info_get(_mesh_if_id, &bbr_info);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
info_ptr->device_count = bbr_info.devices_in_network;
info_ptr->host_timestamp = bbr_info.timestamp;
info_ptr->instance_id = bbr_info.instance_id;
info_ptr->version = bbr_info.version;
memcpy(info_ptr->rpl_dodag_id, bbr_info.dodag_id, 16);
memcpy(info_ptr->ipv6_prefix, bbr_info.prefix, 8);
memcpy(info_ptr->ipv6_iid, bbr_info.IID, 8);
memcpy(info_ptr->gateway_addr, bbr_info.gateway, 16);
return MESH_ERROR_NONE;
}
int WisunBorderRouter::routing_table_get(ws_br_route_info_t *table_ptr, uint16_t table_len)
{
if (table_ptr == NULL) {
return -1;
}
return ws_bbr_routing_table_get(_mesh_if_id, (bbr_route_info_t *)table_ptr, table_len);
}
mesh_error_t WisunBorderRouter::set_radius_server_ipv6_address(const char *address)
{
if (address) {
uint8_t ipv6_addr[16];
if (!stoip6(address, strlen(address), ipv6_addr)) {
return MESH_ERROR_PARAM;
}
// Stored address (returned by get) is in the format given by user of the interface
strcpy(_radius_ipv6_addr, address);
_radius_ipv6_addr_init = true;
_radius_ipv6_addr_set = true;
} else {
_radius_ipv6_addr_init = true;
_radius_ipv6_addr_set = false;
}
return set_bbr_radius_address();
}
mesh_error_t WisunBorderRouter::get_radius_server_ipv6_address(char *address)
{
if (!_radius_ipv6_addr_set) {
return MESH_ERROR_UNKNOWN;
}
strcpy(address, _radius_ipv6_addr);
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::set_bbr_radius_address(void)
{
int status;
if (!_radius_ipv6_addr_init) {
return MESH_ERROR_NONE;
}
if (_radius_ipv6_addr_set) {
uint8_t ipv6_addr[16];
if (!stoip6(_radius_ipv6_addr, strlen(_radius_ipv6_addr), ipv6_addr)) {
return MESH_ERROR_PARAM;
}
status = ws_bbr_radius_address_set(_mesh_if_id, ipv6_addr);
} else {
status = ws_bbr_radius_address_set(_mesh_if_id, NULL);
}
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::set_radius_shared_secret(uint16_t shared_secret_len, const uint8_t *shared_secret)
{
if (shared_secret_len == 0 || !shared_secret) {
return MESH_ERROR_PARAM;
}
if (_shared_secret != NULL) {
delete[] _shared_secret;
}
_shared_secret = new (std::nothrow) char[shared_secret_len];
if (_shared_secret == NULL) {
return MESH_ERROR_MEMORY;
}
memcpy(_shared_secret, shared_secret, shared_secret_len);
_shared_secret_len = shared_secret_len;
int status = ws_bbr_radius_shared_secret_set(_mesh_if_id, shared_secret_len, shared_secret);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return set_bbr_radius_shared_secret();
}
mesh_error_t WisunBorderRouter::set_bbr_radius_shared_secret(void)
{
if (_shared_secret_len == 0 || _shared_secret == NULL) {
return MESH_ERROR_NONE;
}
int status = ws_bbr_radius_shared_secret_set(_mesh_if_id, _shared_secret_len, (uint8_t *) _shared_secret);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::get_radius_shared_secret(uint16_t *shared_secret_len, uint8_t *shared_secret)
{
if (shared_secret_len == NULL) {
return MESH_ERROR_PARAM;
}
int status = ws_bbr_radius_shared_secret_get(_mesh_if_id, shared_secret_len, shared_secret);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::set_radius_timing(ws_br_radius_timing_t *timing)
{
if (timing == NULL) {
return MESH_ERROR_PARAM;
}
if (validate_radius_timing(timing) != MESH_ERROR_NONE) {
return MESH_ERROR_PARAM;
}
_radius_timing = *timing;
_radius_timing_set = true;
return set_bbr_radius_timing();
}
mesh_error_t WisunBorderRouter::set_bbr_radius_timing(void)
{
if (!_radius_timing_set) {
return MESH_ERROR_NONE;
}
bbr_radius_timing_t bbr_timing = {
.radius_retry_imin = _radius_timing.radius_retry_imin,
.radius_retry_imax = _radius_timing.radius_retry_imax,
.radius_retry_count = _radius_timing.radius_retry_count
};
int status = ws_bbr_radius_timing_set(_mesh_if_id, &bbr_timing);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::get_radius_timing(ws_br_radius_timing_t *timing)
{
if (timing == NULL) {
return MESH_ERROR_PARAM;
}
if (_radius_timing_set) {
*timing = _radius_timing;
return MESH_ERROR_NONE;
}
bbr_radius_timing_t bbr_timing;
int status = ws_bbr_radius_timing_get(_mesh_if_id, &bbr_timing);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
timing->radius_retry_imin = bbr_timing.radius_retry_imin;
timing->radius_retry_imax = bbr_timing.radius_retry_imax;
timing->radius_retry_count = bbr_timing.radius_retry_count;
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::validate_radius_timing(ws_br_radius_timing_t *timing)
{
if (timing == NULL) {
return MESH_ERROR_PARAM;
}
bbr_radius_timing_t bbr_timing = {
.radius_retry_imin = timing->radius_retry_imin,
.radius_retry_imax = timing->radius_retry_imax,
.radius_retry_count = timing->radius_retry_count
};
int status = ws_bbr_radius_timing_validate(_mesh_if_id, &bbr_timing);
if (status != 0) {
return MESH_ERROR_UNKNOWN;
}
return MESH_ERROR_NONE;
}
mesh_error_t WisunBorderRouter::set_dns_query_result(SocketAddress *address, char *domain_name)
{
if (!domain_name || !address) {
return MESH_ERROR_PARAM;
}
if (_mesh_if_id < 0) {
return MESH_ERROR_STATE;
}
if (ws_bbr_dns_query_result_set(_mesh_if_id, (const uint8_t *)address->get_ip_bytes(), domain_name) >= 0) {
return MESH_ERROR_NONE;
}
return MESH_ERROR_UNKNOWN;
}