Import Mbed OS hard-float snapshot
This commit is contained in:
@@ -0,0 +1,294 @@
|
||||
/*
|
||||
* Copyright (c) 2017-2018, Arm Limited and affiliates.
|
||||
* 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 "nsconfig.h"
|
||||
#include "string.h"
|
||||
#include "ns_types.h"
|
||||
#include "ns_trace.h"
|
||||
#include "eventOS_scheduler.h"
|
||||
#include "nsdynmemLIB.h"
|
||||
#include "randLIB.h"
|
||||
#include "NWK_INTERFACE/Include/protocol.h"
|
||||
#include "Service_Libs/blacklist/blacklist.h"
|
||||
|
||||
#define TRACE_GROUP "bl"
|
||||
|
||||
static blacklist_entry_t *blacklist_entry_find(const uint8_t *eui64);
|
||||
static void blacklist_entry_add(const uint8_t *eui64);
|
||||
static int8_t blacklist_entry_free(blacklist_entry_t *blacklist_entry);
|
||||
static uint16_t blacklist_entries_count(void);
|
||||
|
||||
static blacklist_data_t *blacklist_data = NULL;
|
||||
|
||||
uint8_t blacklist_init(void)
|
||||
{
|
||||
if (blacklist_data) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
blacklist_data = ns_dyn_mem_alloc(sizeof(blacklist_data_t));
|
||||
|
||||
if (!blacklist_data) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
ns_list_init(&blacklist_data->blacklist);
|
||||
|
||||
blacklist_data->blacklist_purge_ttl = BLACKLIST_DEFAULT_PURGE_TIMER_TIMEOUT;
|
||||
blacklist_data->blacklist_entry_lifetime = BLACKLIST_DEFAULT_ENTRY_LIFETIME;
|
||||
blacklist_data->blacklist_timer_max_timeout = BLACKLIST_DEFAULT_TIMER_MAX_TIMEOUT;
|
||||
blacklist_data->blacklist_timer_timeout = BLACKLIST_DEFAULT_TIMER_TIMEOUT;
|
||||
blacklist_data->blacklist_entry_max_nbr = BLACKLIST_DEFAULT_ENTRY_MAX_NBR;
|
||||
blacklist_data->blacklist_purge_nbr = BLACKLIST_DEFAULT_PURGE_NBR;
|
||||
blacklist_data->blacklist_purge_timer_timeout = BLACKLIST_DEFAULT_PURGE_TIMER_TIMEOUT;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void blacklist_params_set(uint16_t entry_lifetime, uint16_t timer_max_timeout, uint16_t timer_timeout, uint16_t entry_max_nbr, uint16_t purge_nbr, uint16_t purge_timer_timeout)
|
||||
{
|
||||
if (!blacklist_data) {
|
||||
return;
|
||||
}
|
||||
|
||||
blacklist_data->blacklist_purge_ttl = purge_timer_timeout;
|
||||
blacklist_data->blacklist_entry_lifetime = entry_lifetime;
|
||||
blacklist_data->blacklist_timer_max_timeout = timer_max_timeout;
|
||||
blacklist_data->blacklist_timer_timeout = timer_timeout;
|
||||
blacklist_data->blacklist_entry_max_nbr = entry_max_nbr;
|
||||
blacklist_data->blacklist_purge_nbr = purge_nbr;
|
||||
blacklist_data->blacklist_purge_timer_timeout = purge_timer_timeout;
|
||||
}
|
||||
|
||||
bool blacklist_reject(const uint8_t *ll64_address)
|
||||
{
|
||||
if (!blacklist_data) {
|
||||
return false;
|
||||
}
|
||||
|
||||
blacklist_entry_t *blacklist_entry;
|
||||
|
||||
blacklist_entry = blacklist_entry_find(ll64_address + 8);
|
||||
|
||||
// If blacklist entry exists
|
||||
if (blacklist_entry) {
|
||||
// If address is blacklisted rejects
|
||||
if (blacklist_entry->ttl > blacklist_data->blacklist_entry_lifetime) {
|
||||
tr_info("blacklist reject: %s", trace_array(ll64_address + 8, 8));
|
||||
return true;
|
||||
// Neighbor heard; updates blacklist entry TTL to full lifetime
|
||||
} else {
|
||||
blacklist_entry->ttl = blacklist_data->blacklist_entry_lifetime;
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
// If blacklist is full rejects
|
||||
if (blacklist_entries_count() >= blacklist_data->blacklist_entry_max_nbr) {
|
||||
tr_debug("blacklist full reject");
|
||||
return true;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void blacklist_update(const uint8_t *ll64_address, bool success)
|
||||
{
|
||||
if (!blacklist_data) {
|
||||
return;
|
||||
}
|
||||
|
||||
blacklist_entry_t *blacklist_entry;
|
||||
|
||||
if (!ll64_address) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Check that address is LL64 (not LL16)
|
||||
if (memcmp(ll64_address, ADDR_LINK_LOCAL_PREFIX, 8) != 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (memcmp((ll64_address + 8), ADDR_SHORT_ADR_SUFFIC, 6) == 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
blacklist_entry = blacklist_entry_find(ll64_address + 8);
|
||||
|
||||
// On successful link establishment remove address from blacklist
|
||||
if (success) {
|
||||
if (blacklist_entry) {
|
||||
tr_info("Blacklist removed");
|
||||
blacklist_entry_free(blacklist_entry);
|
||||
}
|
||||
// On failure add address to blacklist or update timeout
|
||||
} else {
|
||||
if (blacklist_entry) {
|
||||
blacklist_entry->interval = blacklist_entry->interval * 2;
|
||||
if (blacklist_entry->interval > blacklist_data->blacklist_timer_max_timeout) {
|
||||
blacklist_entry->interval = blacklist_data->blacklist_timer_max_timeout;
|
||||
}
|
||||
/* TTL is blacklist entry lifetime + from 1.0 to 1.5 * interval */
|
||||
blacklist_entry->ttl = blacklist_data->blacklist_entry_lifetime + randLIB_randomise_base(blacklist_entry->interval, 0x8000, 0xC000);
|
||||
} else {
|
||||
tr_info("Blacklist add");
|
||||
blacklist_entry_add(ll64_address + 8);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void blacklist_clear(void)
|
||||
{
|
||||
if (!blacklist_data) {
|
||||
return;
|
||||
}
|
||||
|
||||
ns_list_foreach_safe(blacklist_entry_t, blacklist_entry, &blacklist_data->blacklist) {
|
||||
blacklist_entry_free(blacklist_entry);
|
||||
}
|
||||
}
|
||||
|
||||
void blacklist_free(void)
|
||||
{
|
||||
if (!blacklist_data) {
|
||||
return;
|
||||
}
|
||||
|
||||
ns_dyn_mem_free(blacklist_data);
|
||||
blacklist_data = NULL;
|
||||
}
|
||||
|
||||
void blacklist_ttl_update(uint16_t ticks)
|
||||
{
|
||||
if (!blacklist_data) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (blacklist_data->blacklist_purge_ttl > ticks) {
|
||||
blacklist_data->blacklist_purge_ttl -= ticks;
|
||||
} else {
|
||||
/* 0.5 to 1.5 times timeout */
|
||||
blacklist_data->blacklist_purge_ttl = randLIB_randomise_base(blacklist_data->blacklist_purge_timer_timeout, 0x4000, 0xC000);
|
||||
|
||||
if (blacklist_entries_count() >= blacklist_data->blacklist_entry_max_nbr - blacklist_data->blacklist_purge_nbr) {
|
||||
uint8_t count = 0;
|
||||
blacklist_entry_t *blacklist_entry_min_ttl;
|
||||
|
||||
tr_debug("Blacklist entries purge");
|
||||
|
||||
while (count++ < blacklist_data->blacklist_purge_nbr) {
|
||||
blacklist_entry_min_ttl = NULL;
|
||||
ns_list_foreach(blacklist_entry_t, blacklist_entry, &blacklist_data->blacklist) {
|
||||
if (!blacklist_entry_min_ttl) {
|
||||
blacklist_entry_min_ttl = blacklist_entry;
|
||||
} else if (blacklist_entry->ttl < blacklist_entry_min_ttl->ttl) {
|
||||
blacklist_entry_min_ttl = blacklist_entry;
|
||||
}
|
||||
}
|
||||
if (blacklist_entry_min_ttl) {
|
||||
blacklist_entry_free(blacklist_entry_min_ttl);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
ns_list_foreach_safe(blacklist_entry_t, blacklist_entry, &blacklist_data->blacklist) {
|
||||
if (blacklist_entry->ttl > ticks) {
|
||||
blacklist_entry->ttl -= ticks;
|
||||
} else {
|
||||
tr_info("Blacklist remove entry: %s", trace_array(blacklist_entry->eui64, 8));
|
||||
blacklist_entry_free(blacklist_entry);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static blacklist_entry_t *blacklist_entry_find(const uint8_t *eui64)
|
||||
{
|
||||
if (!eui64) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (!blacklist_data) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
ns_list_foreach(blacklist_entry_t, blacklist_entry, &blacklist_data->blacklist) {
|
||||
if (memcmp(blacklist_entry->eui64, eui64, 8) == 0) {
|
||||
return blacklist_entry;
|
||||
}
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static void blacklist_entry_add(const uint8_t *eui64)
|
||||
{
|
||||
blacklist_entry_t *blacklist_entry;
|
||||
|
||||
if (!eui64) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (!blacklist_data) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (blacklist_entries_count() >= blacklist_data->blacklist_entry_max_nbr) {
|
||||
return;
|
||||
}
|
||||
|
||||
blacklist_entry = ns_dyn_mem_alloc(sizeof(blacklist_entry_t));
|
||||
|
||||
if (!blacklist_entry) {
|
||||
return;
|
||||
}
|
||||
|
||||
blacklist_entry->interval = blacklist_data->blacklist_timer_timeout;
|
||||
|
||||
/* TTL is blacklist entry lifetime + from 1.0 to 1.5 * interval */
|
||||
blacklist_entry->ttl = blacklist_data->blacklist_entry_lifetime + randLIB_randomise_base(blacklist_entry->interval, 0x8000, 0xC000);
|
||||
|
||||
memcpy(blacklist_entry->eui64, eui64, 8);
|
||||
tr_debug("Blacklist add, ttl=%"PRIu16, blacklist_entry->ttl);
|
||||
ns_list_add_to_start(&blacklist_data->blacklist, blacklist_entry);
|
||||
}
|
||||
|
||||
static int8_t blacklist_entry_free(blacklist_entry_t *blacklist_entry)
|
||||
{
|
||||
if (!blacklist_entry) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (!blacklist_data) {
|
||||
return -2;
|
||||
}
|
||||
|
||||
ns_list_remove(&blacklist_data->blacklist, blacklist_entry); //Remove from the list
|
||||
ns_dyn_mem_free(blacklist_entry); // Free entry
|
||||
return 0;
|
||||
}
|
||||
|
||||
static uint16_t blacklist_entries_count(void)
|
||||
{
|
||||
uint16_t entry_count = 0;
|
||||
|
||||
if (!blacklist_data) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
entry_count = ns_list_count(&blacklist_data->blacklist);
|
||||
|
||||
return entry_count;
|
||||
}
|
||||
@@ -0,0 +1,56 @@
|
||||
/*
|
||||
* Copyright (c) 2017, Arm Limited and affiliates.
|
||||
* 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.
|
||||
*/
|
||||
|
||||
#ifndef BLACKLIST_H_
|
||||
#define BLACKLIST_H_
|
||||
|
||||
#define BLACKLIST_DEFAULT_TIMER_TIMEOUT 4
|
||||
#define BLACKLIST_DEFAULT_TIMER_MAX_TIMEOUT 40
|
||||
#define BLACKLIST_DEFAULT_ENTRY_LIFETIME 8
|
||||
#define BLACKLIST_DEFAULT_ENTRY_MAX_NBR 15
|
||||
#define BLACKLIST_DEFAULT_PURGE_NBR 3
|
||||
#define BLACKLIST_DEFAULT_PURGE_TIMER_TIMEOUT 6
|
||||
|
||||
typedef struct blacklist_entry {
|
||||
uint8_t eui64[8];
|
||||
uint16_t ttl; // time to live in is MLE advertisement cycles
|
||||
uint16_t interval; // current interval in MLE advertisement cycles
|
||||
ns_list_link_t link;
|
||||
} blacklist_entry_t;
|
||||
|
||||
typedef NS_LIST_HEAD(blacklist_entry_t, link) blacklist_list_t;
|
||||
|
||||
typedef struct blacklist_data {
|
||||
blacklist_list_t blacklist; // Blacklist entries
|
||||
uint16_t blacklist_purge_ttl; // Blacklist purge ttl
|
||||
uint16_t blacklist_entry_lifetime; // Base lifetime value
|
||||
uint16_t blacklist_timer_max_timeout; // Max. timeout value
|
||||
uint16_t blacklist_timer_timeout; // Initial timeout value
|
||||
uint16_t blacklist_entry_max_nbr; // Max. blacklist entries
|
||||
uint16_t blacklist_purge_nbr; // # of entries to purge
|
||||
uint16_t blacklist_purge_timer_timeout; // Timeout for purge timer
|
||||
} blacklist_data_t;
|
||||
|
||||
uint8_t blacklist_init(void);
|
||||
void blacklist_params_set(uint16_t entry_lifetime, uint16_t timer_max_timeout, uint16_t timer_timeout, uint16_t entry_max_nbr, uint16_t purge_nbr, uint16_t purge_timer_timeout);
|
||||
bool blacklist_reject(const uint8_t *ll64_address);
|
||||
void blacklist_update(const uint8_t *ll64_address, bool success);
|
||||
void blacklist_ttl_update(uint16_t ticks);
|
||||
void blacklist_clear(void);
|
||||
void blacklist_free(void);
|
||||
|
||||
#endif /* BLACKLIST_H_ */
|
||||
Reference in New Issue
Block a user