mbed-os/connectivity/nanostack/sal-stack-nanostack/source/6LoWPAN/IPHC_Decode/lowpan_context.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");
}
}
}