mirror of https://github.com/ARMmbed/mbed-os.git
679 lines
26 KiB
C
679 lines
26 KiB
C
/*
|
|
* Copyright (c) 2018-2019 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 <string.h> //memset
|
|
#include "eventOS_event_timer.h"
|
|
#include "common_functions.h"
|
|
#include "ip6string.h" //ip6tos
|
|
#include "nsdynmemLIB.h"
|
|
#include "include/wisun_tasklet.h"
|
|
#include "include/mesh_system.h"
|
|
#include "ns_event_loop.h"
|
|
#include "fhss_api.h"
|
|
#include "fhss_config.h"
|
|
#include "multicast_api.h"
|
|
#include "mac_api.h"
|
|
#include "sw_mac.h"
|
|
#include "ns_list.h"
|
|
#include "net_interface.h"
|
|
#include "ws_management_api.h" //ws_management_node_init
|
|
#ifdef MBED_CONF_MBED_MESH_API_CERTIFICATE_HEADER
|
|
#if !defined(MBED_CONF_MBED_MESH_API_ROOT_CERTIFICATE) || !defined(MBED_CONF_MBED_MESH_API_OWN_CERTIFICATE) || \
|
|
!defined(MBED_CONF_MBED_MESH_API_OWN_CERTIFICATE_KEY)
|
|
#error Invalid Wi-SUN certificate configuration
|
|
#endif
|
|
#include MBED_CONF_MBED_MESH_API_CERTIFICATE_HEADER
|
|
#endif
|
|
|
|
// For tracing we need to define flag, have include and define group
|
|
//#define HAVE_DEBUG
|
|
#define TRACE_GROUP "WSND"
|
|
#include "ns_trace.h"
|
|
|
|
// Tasklet timer events
|
|
#define TIMER_EVENT_START_BOOTSTRAP 1
|
|
#define INVALID_INTERFACE_ID (-1)
|
|
#define INTERFACE_NAME "WiSunInterface"
|
|
/*
|
|
* Mesh tasklet states.
|
|
*/
|
|
typedef enum {
|
|
TASKLET_STATE_CREATED = 0,
|
|
TASKLET_STATE_INITIALIZED,
|
|
TASKLET_STATE_BOOTSTRAP_STARTED,
|
|
TASKLET_STATE_BOOTSTRAP_FAILED,
|
|
TASKLET_STATE_BOOTSTRAP_READY
|
|
} tasklet_state_t;
|
|
|
|
/*
|
|
* Mesh tasklet data structure.
|
|
*/
|
|
typedef struct {
|
|
void (*mesh_api_cb)(mesh_connection_status_t nwk_status);
|
|
tasklet_state_t tasklet_state;
|
|
int8_t tasklet;
|
|
net_6lowpan_mode_e operating_mode;
|
|
net_6lowpan_mode_extension_e operating_mode_extension;
|
|
int8_t network_interface_id;
|
|
} wisun_tasklet_data_str_t;
|
|
|
|
typedef struct {
|
|
char *network_name;
|
|
uint8_t regulatory_domain;
|
|
uint8_t rd_operating_class;
|
|
uint8_t rd_operating_mode;
|
|
uint8_t uc_channel_function;
|
|
uint8_t bc_channel_function;
|
|
uint8_t uc_dwell_interval;
|
|
uint32_t bc_interval;
|
|
uint8_t bc_dwell_interval;
|
|
uint16_t uc_fixed_channel;
|
|
uint16_t bc_fixed_channel;
|
|
} wisun_network_settings_t;
|
|
|
|
typedef struct {
|
|
arm_certificate_entry_s arm_cert_entry;
|
|
ns_list_link_t link; /*!< List link entry */
|
|
} wisun_certificate_entry_t;
|
|
|
|
typedef NS_LIST_HEAD(wisun_certificate_entry_t, link) cert_list_t;
|
|
typedef struct {
|
|
cert_list_t own_certificates_list;
|
|
cert_list_t trusted_certificates_list;
|
|
bool remove_own_certificates: 1;
|
|
bool remove_trusted_certificates: 1;
|
|
} wisun_certificates_t;
|
|
|
|
#define WS_NA 0xff // Not applicable value
|
|
#define WS_DEFAULT 0x00 // Use default value
|
|
|
|
/* Tasklet data */
|
|
static wisun_tasklet_data_str_t *wisun_tasklet_data_ptr = NULL;
|
|
static wisun_certificates_t *wisun_certificates_ptr = NULL;
|
|
static wisun_network_settings_t wisun_settings_str = {
|
|
.network_name = NULL,
|
|
.regulatory_domain = MBED_CONF_MBED_MESH_API_WISUN_REGULATORY_DOMAIN,
|
|
.rd_operating_class = MBED_CONF_MBED_MESH_API_WISUN_OPERATING_CLASS,
|
|
.rd_operating_mode = MBED_CONF_MBED_MESH_API_WISUN_OPERATING_MODE,
|
|
.uc_channel_function = MBED_CONF_MBED_MESH_API_WISUN_UC_CHANNEL_FUNCTION,
|
|
.bc_channel_function = MBED_CONF_MBED_MESH_API_WISUN_BC_CHANNEL_FUNCTION,
|
|
.uc_dwell_interval = MBED_CONF_MBED_MESH_API_WISUN_UC_DWELL_INTERVAL,
|
|
.bc_interval = MBED_CONF_MBED_MESH_API_WISUN_BC_INTERVAL,
|
|
.bc_dwell_interval = MBED_CONF_MBED_MESH_API_WISUN_BC_DWELL_INTERVAL,
|
|
.uc_fixed_channel = MBED_CONF_MBED_MESH_API_WISUN_UC_FIXED_CHANNEL,
|
|
.bc_fixed_channel = MBED_CONF_MBED_MESH_API_WISUN_BC_FIXED_CHANNEL
|
|
};
|
|
static mac_api_t *mac_api = NULL;
|
|
|
|
extern fhss_timer_t fhss_functions;
|
|
|
|
/* private function prototypes */
|
|
static void wisun_tasklet_main(arm_event_s *event);
|
|
static void wisun_tasklet_network_state_changed(mesh_connection_status_t status);
|
|
static void wisun_tasklet_parse_network_event(arm_event_s *event);
|
|
static void wisun_tasklet_configure_and_connect_to_network(void);
|
|
static void wisun_tasklet_clear_stored_certificates(void) ;
|
|
static int wisun_tasklet_store_certificate_data(const uint8_t *cert, uint16_t cert_len, const uint8_t *cert_key, uint16_t cert_key_len, bool remove_own, bool remove_trusted, bool trusted_cert);
|
|
static int wisun_tasklet_add_stored_certificates(void) ;
|
|
|
|
/*
|
|
* \brief A function which will be eventually called by NanoStack OS when ever the OS has an event to deliver.
|
|
* @param event, describes the sender, receiver and event type.
|
|
*
|
|
* NOTE: Interrupts requested by HW are possible during this function!
|
|
*/
|
|
static void wisun_tasklet_main(arm_event_s *event)
|
|
{
|
|
arm_library_event_type_e event_type;
|
|
event_type = (arm_library_event_type_e) event->event_type;
|
|
|
|
switch (event_type) {
|
|
case ARM_LIB_NWK_INTERFACE_EVENT:
|
|
/* This event is delivered every and each time when there is new
|
|
* information of network connectivity.
|
|
*/
|
|
wisun_tasklet_parse_network_event(event);
|
|
break;
|
|
|
|
case ARM_LIB_TASKLET_INIT_EVENT:
|
|
/* Event with type EV_INIT is an initializer event of NanoStack OS.
|
|
* The event is delivered when the NanoStack OS is running fine.
|
|
* This event should be delivered ONLY ONCE.
|
|
*/
|
|
mesh_system_send_connect_event(wisun_tasklet_data_ptr->tasklet);
|
|
break;
|
|
|
|
case ARM_LIB_SYSTEM_TIMER_EVENT:
|
|
eventOS_event_timer_cancel(event->event_id, wisun_tasklet_data_ptr->tasklet);
|
|
|
|
if (event->event_id == TIMER_EVENT_START_BOOTSTRAP) {
|
|
tr_debug("Restart bootstrap");
|
|
wisun_tasklet_configure_and_connect_to_network();
|
|
}
|
|
break;
|
|
|
|
case APPLICATION_EVENT:
|
|
if (event->event_id == APPL_EVENT_CONNECT) {
|
|
wisun_tasklet_configure_and_connect_to_network();
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
} // switch(event_type)
|
|
}
|
|
|
|
/**
|
|
* \brief Network state event handler.
|
|
* \param event show network start response or current network state.
|
|
*
|
|
* - ARM_NWK_BOOTSTRAP_READY: Save NVK persistent data to NVM and Net role
|
|
* - ARM_NWK_NWK_SCAN_FAIL: Link Layer Active Scan Fail, Stack is Already at Idle state
|
|
* - ARM_NWK_IP_ADDRESS_ALLOCATION_FAIL: No WS Router at current Channel Stack is Already at Idle state
|
|
* - ARM_NWK_NWK_CONNECTION_DOWN: Connection to Access point is lost wait for Scan Result
|
|
* - ARM_NWK_NWK_PARENT_POLL_FAIL: Host should run net start without any PAN-id filter and all channels
|
|
* - ARM_NWK_AUHTENTICATION_FAIL: Pana Authentication fail, Stack is Already at Idle state
|
|
*/
|
|
static void wisun_tasklet_parse_network_event(arm_event_s *event)
|
|
{
|
|
arm_nwk_interface_status_type_e status = (arm_nwk_interface_status_type_e) event->event_data;
|
|
tr_debug("app_parse_network_event() %d", status);
|
|
switch (status) {
|
|
case ARM_NWK_BOOTSTRAP_READY:
|
|
/* Network is ready and node is connected to Access Point */
|
|
if (wisun_tasklet_data_ptr->tasklet_state != TASKLET_STATE_BOOTSTRAP_READY) {
|
|
tr_info("Wi-SUN bootstrap ready");
|
|
wisun_tasklet_data_ptr->tasklet_state = TASKLET_STATE_BOOTSTRAP_READY;
|
|
wisun_tasklet_network_state_changed(MESH_CONNECTED);
|
|
}
|
|
break;
|
|
case ARM_NWK_NWK_SCAN_FAIL:
|
|
/* Link Layer Active Scan Fail, Stack is Already at Idle state */
|
|
tr_debug("Link Layer Scan Fail: No Beacons");
|
|
wisun_tasklet_data_ptr->tasklet_state = TASKLET_STATE_BOOTSTRAP_FAILED;
|
|
wisun_tasklet_network_state_changed(MESH_BOOTSTRAP_FAILED);
|
|
break;
|
|
case ARM_NWK_IP_ADDRESS_ALLOCATION_FAIL:
|
|
/* No WS Router at current Channel Stack is Already at Idle state */
|
|
tr_debug("WS Scan/ GP REG fail");
|
|
wisun_tasklet_data_ptr->tasklet_state = TASKLET_STATE_BOOTSTRAP_FAILED;
|
|
wisun_tasklet_network_state_changed(MESH_BOOTSTRAP_FAILED);
|
|
break;
|
|
case ARM_NWK_NWK_CONNECTION_DOWN:
|
|
/* Connection to Access point is lost wait for Scan Result */
|
|
tr_debug("WS/RPL scan new network");
|
|
wisun_tasklet_data_ptr->tasklet_state = TASKLET_STATE_BOOTSTRAP_FAILED;
|
|
wisun_tasklet_network_state_changed(MESH_BOOTSTRAP_FAILED);
|
|
break;
|
|
case ARM_NWK_NWK_PARENT_POLL_FAIL:
|
|
wisun_tasklet_data_ptr->tasklet_state = TASKLET_STATE_BOOTSTRAP_FAILED;
|
|
wisun_tasklet_network_state_changed(MESH_BOOTSTRAP_FAILED);
|
|
break;
|
|
case ARM_NWK_AUHTENTICATION_FAIL:
|
|
tr_debug("Network authentication fail");
|
|
wisun_tasklet_data_ptr->tasklet_state = TASKLET_STATE_BOOTSTRAP_FAILED;
|
|
wisun_tasklet_network_state_changed(MESH_BOOTSTRAP_FAILED);
|
|
break;
|
|
default:
|
|
tr_warn("Unknown event %d", status);
|
|
break;
|
|
}
|
|
|
|
if (wisun_tasklet_data_ptr->tasklet_state != TASKLET_STATE_BOOTSTRAP_READY &&
|
|
wisun_tasklet_data_ptr->network_interface_id != INVALID_INTERFACE_ID) {
|
|
// Set 5s timer for new network scan
|
|
eventOS_event_timer_request(TIMER_EVENT_START_BOOTSTRAP,
|
|
ARM_LIB_SYSTEM_TIMER_EVENT,
|
|
wisun_tasklet_data_ptr->tasklet,
|
|
5000);
|
|
|
|
}
|
|
}
|
|
|
|
/*
|
|
* \brief Configure and establish network connection
|
|
*
|
|
*/
|
|
static void wisun_tasklet_configure_and_connect_to_network(void)
|
|
{
|
|
int status;
|
|
fhss_timer_t *fhss_timer_ptr = &fhss_functions;
|
|
|
|
wisun_tasklet_data_ptr->operating_mode = NET_6LOWPAN_ROUTER;
|
|
wisun_tasklet_data_ptr->operating_mode_extension = NET_6LOWPAN_WS;
|
|
|
|
arm_nwk_interface_configure_6lowpan_bootstrap_set(
|
|
wisun_tasklet_data_ptr->network_interface_id,
|
|
wisun_tasklet_data_ptr->operating_mode,
|
|
wisun_tasklet_data_ptr->operating_mode_extension);
|
|
|
|
if (wisun_tasklet_add_stored_certificates() != 0) {
|
|
tr_error("Can't set Wi-SUN certificates");
|
|
return;
|
|
}
|
|
|
|
status = ws_management_node_init(wisun_tasklet_data_ptr->network_interface_id,
|
|
MBED_CONF_MBED_MESH_API_WISUN_REGULATORY_DOMAIN,
|
|
wisun_settings_str.network_name,
|
|
fhss_timer_ptr);
|
|
if (status < 0) {
|
|
tr_error("Failed to initialize WS");
|
|
return;
|
|
}
|
|
|
|
if (wisun_settings_str.uc_channel_function != WS_NA) {
|
|
status = ws_management_fhss_unicast_channel_function_configure(wisun_tasklet_data_ptr->network_interface_id,
|
|
wisun_settings_str.uc_channel_function,
|
|
wisun_settings_str.uc_fixed_channel,
|
|
wisun_settings_str.uc_dwell_interval);
|
|
|
|
if (status < 0) {
|
|
tr_error("Failed to set unicast channel function configuration");
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (wisun_settings_str.bc_channel_function != WS_NA ||
|
|
wisun_settings_str.bc_dwell_interval != WS_DEFAULT ||
|
|
wisun_settings_str.bc_interval != WS_DEFAULT) {
|
|
status = ws_management_fhss_broadcast_channel_function_configure(wisun_tasklet_data_ptr->network_interface_id,
|
|
wisun_settings_str.bc_channel_function,
|
|
wisun_settings_str.bc_fixed_channel,
|
|
wisun_settings_str.bc_dwell_interval,
|
|
wisun_settings_str.bc_interval);
|
|
|
|
if (status < 0) {
|
|
tr_error("Failed to set broadcast channel function configuration");
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (wisun_settings_str.uc_dwell_interval != WS_DEFAULT ||
|
|
wisun_settings_str.bc_dwell_interval != WS_DEFAULT ||
|
|
wisun_settings_str.bc_interval != WS_DEFAULT) {
|
|
status = ws_management_fhss_timing_configure(wisun_tasklet_data_ptr->network_interface_id,
|
|
wisun_settings_str.uc_dwell_interval,
|
|
wisun_settings_str.bc_interval,
|
|
wisun_settings_str.bc_dwell_interval);
|
|
|
|
if (status < 0) {
|
|
tr_error("Failed to set fhss configuration");
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (wisun_settings_str.regulatory_domain != WS_NA ||
|
|
wisun_settings_str.rd_operating_class != WS_NA ||
|
|
wisun_settings_str.rd_operating_mode != WS_NA) {
|
|
status = ws_management_regulatory_domain_set(wisun_tasklet_data_ptr->network_interface_id,
|
|
wisun_settings_str.regulatory_domain,
|
|
wisun_settings_str.rd_operating_class,
|
|
wisun_settings_str.rd_operating_mode);
|
|
|
|
if (status < 0) {
|
|
tr_error("Failed to set regulatory domain!");
|
|
return;
|
|
}
|
|
}
|
|
|
|
#if defined(MBED_CONF_MBED_MESH_API_CERTIFICATE_HEADER)
|
|
arm_certificate_entry_s trusted_cert = {
|
|
.cert = MBED_CONF_MBED_MESH_API_ROOT_CERTIFICATE,
|
|
.key = NULL,
|
|
.cert_len = 0,
|
|
.key_len = 0
|
|
};
|
|
#ifdef MBED_CONF_MBED_MESH_API_ROOT_CERTIFICATE_LEN
|
|
trusted_cert.cert_len = MBED_CONF_MBED_MESH_API_ROOT_CERTIFICATE_LEN;
|
|
#else
|
|
trusted_cert.cert_len = strlen((const char *) MBED_CONF_MBED_MESH_API_ROOT_CERTIFICATE) + 1;
|
|
#endif
|
|
arm_network_trusted_certificate_add((const arm_certificate_entry_s *)&trusted_cert);
|
|
|
|
arm_certificate_entry_s own_cert = {
|
|
.cert = MBED_CONF_MBED_MESH_API_OWN_CERTIFICATE,
|
|
.key = MBED_CONF_MBED_MESH_API_OWN_CERTIFICATE_KEY,
|
|
.cert_len = 0,
|
|
.key_len = 0
|
|
};
|
|
#ifdef MBED_CONF_MBED_MESH_API_OWN_CERTIFICATE_LEN
|
|
own_cert.cert_len = MBED_CONF_MBED_MESH_API_OWN_CERTIFICATE_LEN;
|
|
#else
|
|
own_cert.cert_len = strlen((const char *) MBED_CONF_MBED_MESH_API_OWN_CERTIFICATE) + 1;
|
|
#endif
|
|
#ifdef MBED_CONF_MBED_MESH_API_OWN_CERTIFICATE_KEY_LEN
|
|
own_cert.key_len = MBED_CONF_MBED_MESH_API_OWN_CERTIFICATE_KEY_LEN;
|
|
#else
|
|
own_cert.key_len = strlen((const char *) MBED_CONF_MBED_MESH_API_OWN_CERTIFICATE_KEY) + 1;
|
|
#endif
|
|
arm_network_own_certificate_add((const arm_certificate_entry_s *)&own_cert);
|
|
#endif
|
|
|
|
status = arm_nwk_interface_up(wisun_tasklet_data_ptr->network_interface_id);
|
|
if (status >= 0) {
|
|
wisun_tasklet_data_ptr->tasklet_state = TASKLET_STATE_BOOTSTRAP_STARTED;
|
|
tr_info("Start Wi-SUN Bootstrap");
|
|
wisun_tasklet_network_state_changed(MESH_BOOTSTRAP_STARTED);
|
|
} else {
|
|
wisun_tasklet_data_ptr->tasklet_state = TASKLET_STATE_BOOTSTRAP_FAILED;
|
|
tr_err("Bootstrap start failed, %d", status);
|
|
wisun_tasklet_network_state_changed(MESH_BOOTSTRAP_START_FAILED);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Inform application about network state change
|
|
*/
|
|
static void wisun_tasklet_network_state_changed(mesh_connection_status_t status)
|
|
{
|
|
if (wisun_tasklet_data_ptr->mesh_api_cb) {
|
|
(wisun_tasklet_data_ptr->mesh_api_cb)(status);
|
|
}
|
|
}
|
|
|
|
static int wisun_tasklet_store_certificate_data(const uint8_t *cert, uint16_t cert_len, const uint8_t *cert_key, uint16_t cert_key_len, bool remove_own, bool remove_trusted, bool trusted_cert)
|
|
{
|
|
if (wisun_certificates_ptr == NULL) {
|
|
wisun_certificates_ptr = (wisun_certificates_t *)ns_dyn_mem_alloc(sizeof(wisun_certificates_t));
|
|
if (!wisun_certificates_ptr) {
|
|
return -1;
|
|
}
|
|
ns_list_init(&wisun_certificates_ptr->own_certificates_list);
|
|
ns_list_init(&wisun_certificates_ptr->trusted_certificates_list);
|
|
wisun_certificates_ptr->remove_own_certificates = false;
|
|
wisun_certificates_ptr->remove_trusted_certificates = false;
|
|
}
|
|
|
|
if (remove_own) {
|
|
wisun_certificates_ptr->remove_own_certificates = true;
|
|
return 0;
|
|
}
|
|
|
|
if (remove_trusted) {
|
|
wisun_certificates_ptr->remove_trusted_certificates = true;
|
|
return 0;
|
|
}
|
|
|
|
wisun_certificate_entry_t *ws_cert_entry_store = (wisun_certificate_entry_t *)ns_dyn_mem_alloc(sizeof(wisun_certificate_entry_t));
|
|
if (!ws_cert_entry_store) {
|
|
wisun_tasklet_clear_stored_certificates();
|
|
return -1;
|
|
}
|
|
|
|
ws_cert_entry_store->arm_cert_entry.cert = cert;
|
|
ws_cert_entry_store->arm_cert_entry.cert_len = cert_len;
|
|
|
|
if (cert_key) {
|
|
ws_cert_entry_store->arm_cert_entry.key = cert_key;
|
|
ws_cert_entry_store->arm_cert_entry.key_len = cert_key_len;
|
|
} else {
|
|
ws_cert_entry_store->arm_cert_entry.key = NULL;
|
|
ws_cert_entry_store->arm_cert_entry.key_len = 0;
|
|
}
|
|
|
|
if (trusted_cert) {
|
|
ns_list_add_to_end(&wisun_certificates_ptr->trusted_certificates_list, ws_cert_entry_store);
|
|
} else {
|
|
ns_list_add_to_end(&wisun_certificates_ptr->own_certificates_list, ws_cert_entry_store);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void wisun_tasklet_clear_stored_certificates(void)
|
|
{
|
|
if (!wisun_certificates_ptr) {
|
|
return;
|
|
}
|
|
|
|
ns_list_foreach_safe(wisun_certificate_entry_t, trusted_cert_entry, &wisun_certificates_ptr->trusted_certificates_list) {
|
|
ns_list_remove(&wisun_certificates_ptr->trusted_certificates_list, trusted_cert_entry);
|
|
ns_dyn_mem_free(trusted_cert_entry);
|
|
}
|
|
|
|
ns_list_foreach_safe(wisun_certificate_entry_t, own_cert_entry, &wisun_certificates_ptr->own_certificates_list) {
|
|
ns_list_remove(&wisun_certificates_ptr->own_certificates_list, own_cert_entry);
|
|
ns_dyn_mem_free(own_cert_entry);
|
|
}
|
|
|
|
ns_dyn_mem_free(wisun_certificates_ptr);
|
|
wisun_certificates_ptr = NULL;
|
|
}
|
|
|
|
static int wisun_tasklet_add_stored_certificates(void)
|
|
{
|
|
int8_t status = 0;
|
|
|
|
if (wisun_certificates_ptr == NULL) {
|
|
// certificates not updated
|
|
return 0;
|
|
}
|
|
|
|
if (wisun_certificates_ptr->remove_own_certificates) {
|
|
status = arm_network_own_certificates_remove();
|
|
if (status != 0) {
|
|
goto CERTIFICATE_SET_END;
|
|
}
|
|
}
|
|
|
|
if (wisun_certificates_ptr->remove_trusted_certificates) {
|
|
status = arm_network_trusted_certificates_remove();
|
|
if (status != 0) {
|
|
goto CERTIFICATE_SET_END;
|
|
}
|
|
}
|
|
|
|
ns_list_foreach(wisun_certificate_entry_t, cert_entry, &wisun_certificates_ptr->trusted_certificates_list) {
|
|
status = arm_network_trusted_certificate_add(&cert_entry->arm_cert_entry);
|
|
if (status != 0) {
|
|
goto CERTIFICATE_SET_END;
|
|
}
|
|
}
|
|
|
|
ns_list_foreach(wisun_certificate_entry_t, cert_entry, &wisun_certificates_ptr->own_certificates_list) {
|
|
status = arm_network_own_certificate_add(&cert_entry->arm_cert_entry);
|
|
if (status != 0) {
|
|
goto CERTIFICATE_SET_END;
|
|
}
|
|
}
|
|
|
|
CERTIFICATE_SET_END:
|
|
wisun_tasklet_clear_stored_certificates();
|
|
|
|
return status;
|
|
}
|
|
|
|
/* Public functions */
|
|
int8_t wisun_tasklet_get_router_ip_address(char *address, int8_t len)
|
|
{
|
|
network_layer_address_s nd_address;
|
|
|
|
if ((len >= 40) && (0 == arm_nwk_nd_address_read(wisun_tasklet_data_ptr->network_interface_id, &nd_address))) {
|
|
ip6tos(nd_address.border_router, address);
|
|
tr_debug("Router IP address: %s", address);
|
|
return 0;
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
int8_t wisun_tasklet_connect(mesh_interface_cb callback, int8_t nwk_interface_id)
|
|
{
|
|
bool re_connecting = true;
|
|
int8_t tasklet_id = wisun_tasklet_data_ptr->tasklet;
|
|
|
|
if (wisun_tasklet_data_ptr->network_interface_id != INVALID_INTERFACE_ID) {
|
|
return -3; // already connected to network
|
|
}
|
|
|
|
if (wisun_tasklet_data_ptr->tasklet_state == TASKLET_STATE_CREATED) {
|
|
re_connecting = false;
|
|
}
|
|
|
|
memset(wisun_tasklet_data_ptr, 0, sizeof(wisun_tasklet_data_str_t));
|
|
wisun_tasklet_data_ptr->mesh_api_cb = callback;
|
|
wisun_tasklet_data_ptr->network_interface_id = nwk_interface_id;
|
|
wisun_tasklet_data_ptr->tasklet_state = TASKLET_STATE_INITIALIZED;
|
|
|
|
if (re_connecting == false) {
|
|
wisun_tasklet_data_ptr->tasklet = eventOS_event_handler_create(&wisun_tasklet_main,
|
|
ARM_LIB_TASKLET_INIT_EVENT);
|
|
if (wisun_tasklet_data_ptr->tasklet < 0) {
|
|
// -1 handler already used by other tasklet
|
|
// -2 memory allocation failure
|
|
return wisun_tasklet_data_ptr->tasklet;
|
|
}
|
|
} else {
|
|
wisun_tasklet_data_ptr->tasklet = tasklet_id;
|
|
mesh_system_send_connect_event(wisun_tasklet_data_ptr->tasklet);
|
|
}
|
|
|
|
return wisun_tasklet_data_ptr->tasklet;
|
|
}
|
|
|
|
int8_t wisun_tasklet_disconnect(bool send_cb)
|
|
{
|
|
int8_t status = -1;
|
|
if (wisun_tasklet_data_ptr != NULL) {
|
|
if (wisun_tasklet_data_ptr->network_interface_id != INVALID_INTERFACE_ID) {
|
|
status = arm_nwk_interface_down(wisun_tasklet_data_ptr->network_interface_id);
|
|
wisun_tasklet_data_ptr->network_interface_id = INVALID_INTERFACE_ID;
|
|
if (send_cb == true) {
|
|
wisun_tasklet_network_state_changed(MESH_DISCONNECTED);
|
|
}
|
|
}
|
|
wisun_tasklet_data_ptr->mesh_api_cb = NULL;
|
|
}
|
|
return status;
|
|
}
|
|
|
|
void wisun_tasklet_init(void)
|
|
{
|
|
if (wisun_tasklet_data_ptr == NULL) {
|
|
wisun_tasklet_data_ptr = (wisun_tasklet_data_str_t *)ns_dyn_mem_alloc(sizeof(wisun_tasklet_data_str_t));
|
|
// allocation not validated, in case of failure execution stops here
|
|
memset(wisun_tasklet_data_ptr, 0, sizeof(wisun_tasklet_data_str_t));
|
|
wisun_tasklet_data_ptr->tasklet_state = TASKLET_STATE_CREATED;
|
|
wisun_tasklet_data_ptr->network_interface_id = INVALID_INTERFACE_ID;
|
|
}
|
|
}
|
|
|
|
int8_t wisun_tasklet_network_init(int8_t device_id)
|
|
{
|
|
// TODO, read interface name from configuration
|
|
mac_description_storage_size_t storage_sizes;
|
|
storage_sizes.device_decription_table_size = 32;
|
|
storage_sizes.key_description_table_size = 4;
|
|
storage_sizes.key_lookup_size = 1;
|
|
storage_sizes.key_usage_size = 3;
|
|
if (!mac_api) {
|
|
mac_api = ns_sw_mac_create(device_id, &storage_sizes);
|
|
}
|
|
|
|
if (!wisun_settings_str.network_name) {
|
|
// No network name set by API, use network name from configuration
|
|
int wisun_network_name_len = sizeof(MBED_CONF_MBED_MESH_API_WISUN_NETWORK_NAME);
|
|
wisun_settings_str.network_name = (char *)ns_dyn_mem_alloc(wisun_network_name_len);
|
|
if (!wisun_settings_str.network_name) {
|
|
return -3;
|
|
}
|
|
strncpy(wisun_settings_str.network_name, MBED_CONF_MBED_MESH_API_WISUN_NETWORK_NAME, wisun_network_name_len);
|
|
}
|
|
|
|
return arm_nwk_interface_lowpan_init(mac_api, INTERFACE_NAME);
|
|
}
|
|
|
|
int wisun_tasklet_set_network_name(int8_t nwk_interface_id, char *network_name_ptr)
|
|
{
|
|
if (!network_name_ptr || strlen(network_name_ptr) > 32) {
|
|
return -1;
|
|
}
|
|
|
|
// save the network name to have support for disconnect/connect
|
|
ns_dyn_mem_free(wisun_settings_str.network_name);
|
|
wisun_settings_str.network_name = (char *)ns_dyn_mem_alloc(strlen(network_name_ptr) + 1);
|
|
if (!wisun_settings_str.network_name) {
|
|
return -2;
|
|
}
|
|
|
|
strcpy(wisun_settings_str.network_name, network_name_ptr);
|
|
|
|
if (wisun_tasklet_data_ptr && wisun_tasklet_data_ptr->tasklet_state == TASKLET_STATE_BOOTSTRAP_READY) {
|
|
// interface is up, try to change name dynamically
|
|
return ws_management_network_name_set(nwk_interface_id, wisun_settings_str.network_name);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int wisun_tasklet_set_regulatory_domain(int8_t nwk_interface_id, uint8_t regulatory_domain, uint8_t operating_class, uint8_t operating_mode)
|
|
{
|
|
int status = 0;
|
|
|
|
wisun_settings_str.regulatory_domain = regulatory_domain;
|
|
wisun_settings_str.rd_operating_class = operating_class;
|
|
wisun_settings_str.rd_operating_mode = operating_mode;
|
|
|
|
if (wisun_tasklet_data_ptr && wisun_tasklet_data_ptr->tasklet_state == TASKLET_STATE_BOOTSTRAP_READY) {
|
|
status = ws_management_regulatory_domain_set(nwk_interface_id, regulatory_domain, operating_class, operating_mode);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
int wisun_tasklet_set_own_certificate(uint8_t *cert, uint16_t cert_len, uint8_t *cert_key, uint16_t cert_key_len)
|
|
{
|
|
if (wisun_tasklet_data_ptr) {
|
|
// this API can be only used before first connect()
|
|
tr_err("Already connected");
|
|
return -2;
|
|
}
|
|
|
|
return wisun_tasklet_store_certificate_data(cert, cert_len, cert_key, cert_key_len, false, false, false);
|
|
}
|
|
|
|
int wisun_tasklet_remove_own_certificates(void)
|
|
{
|
|
if (wisun_tasklet_data_ptr) {
|
|
// this API can be only used before first connect()
|
|
tr_err("Already connected");
|
|
return -2;
|
|
}
|
|
|
|
return wisun_tasklet_store_certificate_data(NULL, 0, NULL, 0, true, false, false);
|
|
}
|
|
|
|
int wisun_tasklet_remove_trusted_certificates(void)
|
|
{
|
|
if (wisun_tasklet_data_ptr) {
|
|
// this API can be only used before first connect()
|
|
tr_err("Already connected");
|
|
return -2;
|
|
}
|
|
|
|
return wisun_tasklet_store_certificate_data(NULL, 0, NULL, 0, false, true, false);
|
|
}
|
|
|
|
int wisun_tasklet_set_trusted_certificate(uint8_t *cert, uint16_t cert_len)
|
|
{
|
|
if (wisun_tasklet_data_ptr) {
|
|
// this API can be only used before first connect()
|
|
tr_err("Already connected");
|
|
return -2;
|
|
}
|
|
return wisun_tasklet_store_certificate_data(cert, cert_len, NULL, 0, false, false, true);
|
|
}
|