2019-03-26 11:39:29 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 Freie Universität Berlin
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup ble_nimble_scanlist
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Implementation of a list for NimBLE scan results
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
#include <limits.h>
|
|
|
|
|
|
|
|
#include "xtimer.h"
|
|
|
|
#include "net/bluetil/ad.h"
|
|
|
|
|
|
|
|
#include "nimble_scanlist.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
static nimble_scanlist_entry_t _mem[NIMBLE_SCANLIST_SIZE];
|
|
|
|
|
|
|
|
static clist_node_t _pool;
|
|
|
|
static clist_node_t _list;
|
|
|
|
|
|
|
|
static int _finder(clist_node_t *node, void *arg)
|
|
|
|
{
|
|
|
|
const ble_addr_t *addr = (const ble_addr_t *)arg;
|
|
|
|
nimble_scanlist_entry_t *e = (nimble_scanlist_entry_t *)node;
|
|
|
|
if (ble_addr_cmp(&e->addr, addr) == 0) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static nimble_scanlist_entry_t *_find(const ble_addr_t *addr)
|
|
|
|
{
|
|
|
|
return (nimble_scanlist_entry_t *)clist_foreach(&_list, _finder,
|
|
|
|
(void *)addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void nimble_scanlist_init(void)
|
|
|
|
{
|
2019-07-18 15:20:42 +02:00
|
|
|
for (unsigned i = 0; i < ARRAY_SIZE(_mem); i++) {
|
2019-03-26 11:39:29 +01:00
|
|
|
clist_rpush(&_pool, &_mem[i].node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void nimble_scanlist_update(const ble_addr_t *addr, int8_t rssi,
|
|
|
|
const uint8_t *ad, size_t len)
|
|
|
|
{
|
|
|
|
assert(addr);
|
|
|
|
assert(len <= BLE_ADV_PDU_LEN);
|
|
|
|
|
|
|
|
uint32_t now = xtimer_now_usec();
|
|
|
|
nimble_scanlist_entry_t *e = _find(addr);
|
|
|
|
|
|
|
|
if (!e) {
|
|
|
|
e = (nimble_scanlist_entry_t *)clist_lpop(&_pool);
|
|
|
|
if (!e) {
|
|
|
|
/* no space available, dropping newly discovered node */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
memcpy(&e->addr, addr, sizeof(ble_addr_t));
|
|
|
|
if (ad) {
|
|
|
|
memcpy(e->ad, ad, len);
|
|
|
|
}
|
|
|
|
e->ad_len = len;
|
|
|
|
e->last_rssi = rssi;
|
|
|
|
e->first_update = now;
|
|
|
|
e->adv_msg_cnt = 1;
|
|
|
|
clist_rpush(&_list, (clist_node_t *)e);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
e->adv_msg_cnt++;
|
|
|
|
}
|
|
|
|
|
|
|
|
e->last_update = now;
|
|
|
|
}
|
|
|
|
|
|
|
|
nimble_scanlist_entry_t *nimble_scanlist_get_next(nimble_scanlist_entry_t *e)
|
|
|
|
{
|
|
|
|
if (e == NULL) {
|
|
|
|
return (nimble_scanlist_entry_t *)_list.next;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
e = (nimble_scanlist_entry_t *)e->node.next;
|
|
|
|
if (e == (nimble_scanlist_entry_t *)_list.next) {
|
|
|
|
/* end of list */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
nimble_scanlist_entry_t *nimble_scanlist_get(unsigned pos)
|
|
|
|
{
|
|
|
|
nimble_scanlist_entry_t *cur = nimble_scanlist_get_next(NULL);
|
|
|
|
for (unsigned i = 0; i < pos; i++) {
|
|
|
|
cur = nimble_scanlist_get_next(cur);
|
|
|
|
}
|
|
|
|
return cur;
|
|
|
|
}
|
|
|
|
|
|
|
|
void nimble_scanlist_clear(void)
|
|
|
|
{
|
|
|
|
clist_node_t *node = clist_lpop(&_list);
|
|
|
|
while (node) {
|
|
|
|
clist_rpush(&_pool, node);
|
|
|
|
node = clist_lpop(&_list);
|
|
|
|
}
|
|
|
|
}
|