mirror of https://github.com/ARMmbed/mbed-os.git
152 lines
4.5 KiB
C
152 lines
4.5 KiB
C
/*
|
|
* Copyright (c) 2015-2017, 2020, Pelion 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.
|
|
*/
|
|
|
|
/*
|
|
* \file lowpan_context.c
|
|
* \brief API for Add,Remove and update timeouts for lowpan context's
|
|
*
|
|
*/
|
|
#include "nsconfig.h"
|
|
#include "string.h"
|
|
#include "ns_types.h"
|
|
#include "ns_trace.h"
|
|
#include "nsdynmemLIB.h"
|
|
#include "ns_list.h"
|
|
#include "6LoWPAN/IPHC_Decode/lowpan_context.h"
|
|
#include "common_functions.h"
|
|
|
|
#define TRACE_GROUP "lCon"
|
|
|
|
lowpan_context_t *lowpan_context_get_by_id(const lowpan_context_list_t *list, uint8_t id)
|
|
{
|
|
id &= LOWPAN_CONTEXT_CID_MASK;
|
|
/* Check to see we already have info for this context */
|
|
ns_list_foreach(lowpan_context_t, entry, list) {
|
|
if (entry->cid == id) {
|
|
return entry;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
lowpan_context_t *lowpan_context_get_by_address(const lowpan_context_list_t *list, const uint8_t *ipv6Address)
|
|
{
|
|
/* Check to see we already have info for this context
|
|
* List is already listed that longest prefix are first at list
|
|
*/
|
|
ns_list_foreach(lowpan_context_t, entry, list) {
|
|
if (bitsequal(entry->prefix, ipv6Address, entry->length)) {
|
|
//Take always longest match prefix
|
|
return entry;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
int_fast8_t lowpan_context_update(lowpan_context_list_t *list, uint8_t cid_flags, uint16_t lifetime, const uint8_t *prefix, uint_fast8_t len, bool stable)
|
|
{
|
|
uint8_t cid = cid_flags & LOWPAN_CONTEXT_CID_MASK;
|
|
lowpan_context_t *ctx = NULL;
|
|
|
|
/* Check to see we already have info for this context */
|
|
|
|
ctx = lowpan_context_get_by_id(list, cid);
|
|
if (ctx) {
|
|
//Remove from the list - it will be reinserted below, sorted by its
|
|
//new context length. (Don't need "safe" foreach, as we break
|
|
//immediately after the removal).
|
|
ns_list_remove(list, ctx);
|
|
}
|
|
|
|
if (lifetime == 0) {
|
|
/* This is a removal request: delete any existing entry, then exit */
|
|
if (ctx) {
|
|
ns_dyn_mem_free(ctx);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
if (!ctx) {
|
|
ctx = ns_dyn_mem_alloc(sizeof(lowpan_context_t));
|
|
}
|
|
|
|
if (!ctx) {
|
|
tr_error("No heap for New 6LoWPAN Context");
|
|
return -2;
|
|
}
|
|
|
|
bool inserted = false;
|
|
ns_list_foreach(lowpan_context_t, entry, list) {
|
|
if (len >= entry->length) {
|
|
ns_list_add_before(list, entry, ctx);
|
|
inserted = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!inserted) {
|
|
ns_list_add_to_end(list, ctx);
|
|
}
|
|
|
|
ctx->length = len;
|
|
ctx->cid = cid;
|
|
ctx->expiring = false;
|
|
ctx->stable = stable;
|
|
ctx->compression = cid_flags & LOWPAN_CONTEXT_C;
|
|
ctx->lifetime = (uint32_t) lifetime * 600u; /* minutes -> 100ms ticks */
|
|
|
|
// Do our own zero-padding, just in case sender has done something weird
|
|
memset(ctx->prefix, 0, sizeof ctx->prefix);
|
|
bitcopy(ctx->prefix, prefix, len);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void lowpan_context_list_free(lowpan_context_list_t *list)
|
|
{
|
|
ns_list_foreach_safe(lowpan_context_t, cur, list) {
|
|
ns_list_remove(list, cur);
|
|
ns_dyn_mem_free(cur);
|
|
}
|
|
}
|
|
|
|
/* ticks is in 1/10s */
|
|
void lowpan_context_timer(lowpan_context_list_t *list, uint_fast16_t ticks)
|
|
{
|
|
ns_list_foreach_safe(lowpan_context_t, ctx, list) {
|
|
if (ctx->lifetime > ticks) {
|
|
ctx->lifetime -= ticks;
|
|
continue;
|
|
}
|
|
if (!ctx->expiring) {
|
|
/* Main lifetime has run out. Clear compression flag, and retain a
|
|
* bit longer (RFC 6775 5.4.3).
|
|
*/
|
|
ctx->compression = false;
|
|
ctx->expiring = true;
|
|
ctx->lifetime = 2 * 18000u; /* 2 * default Router Lifetime = 2 * 1800s = 1 hour */
|
|
tr_debug("Context timed out - compression disabled");
|
|
} else {
|
|
/* 1-hour expiration timer set above has run out */
|
|
ns_list_remove(list, ctx);
|
|
ns_dyn_mem_free(ctx);
|
|
tr_debug("Delete Expired context");
|
|
}
|
|
}
|
|
}
|
|
|