· 

Bluetooth AoA 複数台の送信

NoerdicのSampleは1台受信に対し1台送信しか行えないことがわかり、

1台受信に対し複数送信が行えないかを調べてみました。

環境

環境等は前回のブログBluetooth AoA に

Direction finding connectionless beacon (送信側 nRF52833DK 1台を使用)を追加します。(全く同じもの1台追加) 

修正箇所

Direction finding connectionless locator側のみです。

修正ファイルは

  • main.c
  • prj.conf

動作流れは

  • AoAデバイス識別の宣言配列化
  • アドバタイズスキャン開始
  • アドバタイズスキャンでAoA識別しできたデバイスを配列に保管
  • 同じものが2度見つかったらアドバタイズスキャン終了
  • 配列に保管したデバイスをcreate_sync()とenable_cte_rx()で初期化と実行

となります。

ただ、アドバタイズスキャンの再開始のタイミングが全てのデバイスの消失となっていて、

デバイスの動的追加はできません。(ココはケースに合わせて実装でしょうか)

あと、prj.confに「CONFIG_BT_PER_ADV_SYNC_MAX=7」を1行追加している理由ですが

これは

 

  • test\locator\zephyr\subsys\bluetooth\host\scan.cの
    • per_adv_sync_new()
    • コーススタック
      • main()
        • create_sync()
          • bt_le_per_adv_sync_create()
            • per_adv_sync_new()

にて

  • for (int i = 0; i < ARRAY_SIZE(per_adv_sync_pool); i++)

となっていてkconfig.advのCONFIG_BT_PER_ADV_SYNC_MAXのdefaultが1となっているため、

1台しか使えなくなっているためになります。

ただCONFIG_BT_PER_ADV_SYNC_MAXを増やすとramの消費が激しく、今回は7が限界でした。。。

 


/////////  main.c /////////// 

/*

 * Copyright (c) 2021 Nordic Semiconductor ASA

 *

 * SPDX-License-Identifier: LicenseRef-Nordic-5-Clause

 */

 

#include <stddef.h>

#include <errno.h>

#include <zephyr/kernel.h>

 

#include <zephyr/bluetooth/bluetooth.h>

#include <zephyr/bluetooth/direction.h>

 

#define DEVICE_NAME CONFIG_BT_DEVICE_NAME

#define DEVICE_NAME_LEN (sizeof(DEVICE_NAME) - 1)

#define PEER_NAME_LEN_MAX 30

/* BT Core 5.3 specification allows controller to wait 6 periodic advertising events for

 * synchronization establishment, hence timeout must be longer than that.

 */

#define SYNC_CREATE_TIMEOUT_INTERVAL_NUM 7

/* Maximum length of advertising data represented in hexadecimal format */

#define ADV_DATA_HEX_STR_LEN_MAX (BT_GAP_ADV_MAX_EXT_ADV_DATA_LEN * 2 + 1)

 

static bool scan_enabled;

// static struct bt_le_per_adv_sync *sync;

// static bt_addr_le_t per_addr;

// static volatile bool per_adv_found;

// static uint8_t per_sid;

// static uint32_t sync_create_timeout_ms;

 

#define MAX_BEACON_DEVICE 10

static char name_buf[MAX_BEACON_DEVICE][PEER_NAME_LEN_MAX];

static struct bt_le_per_adv_sync *sync[PEER_NAME_LEN_MAX];

static bt_addr_le_t per_addr[MAX_BEACON_DEVICE];

static volatile bool per_adv_found;

static uint8_t per_sid[MAX_BEACON_DEVICE];

static uint32_t sync_create_timeout_ms[MAX_BEACON_DEVICE];

static uint8_t device_no;

 

 

static K_SEM_DEFINE(sem_per_adv, 0, 1);

static K_SEM_DEFINE(sem_per_sync, 0, 1);

static K_SEM_DEFINE(sem_per_sync_lost, 0, 1);

 

#if defined(CONFIG_BT_DF_CTE_RX_AOA)

/* Example sequence of antenna switch patterns for antenna matrix designed by

 * Nordic. For more information about antenna switch patterns see README.rst.

 */

static const uint8_t ant_patterns[] = { 0x2, 0x0, 0x5, 0x6, 0x1, 0x4,

0xC, 0x9, 0xE, 0xD, 0x8, 0xA };

// static const uint8_t ant_patterns[] = { 0x2, 0x0, 0x5};

#endif /* CONFIG_BT_DF_CTE_RX_AOA */

 

static inline uint32_t adv_interval_to_ms(uint16_t interval)

{

return interval * 5 / 4;

}

 

static const char *phy2str(uint8_t phy)

{

switch (phy) {

case 0: return "No packets";

case BT_GAP_LE_PHY_1M: return "LE 1M";

case BT_GAP_LE_PHY_2M: return "LE 2M";

case BT_GAP_LE_PHY_CODED: return "LE Coded";

default: return "Unknown";

}

}

 

static const char *cte_type2str(uint8_t type)

{

switch (type) {

case BT_DF_CTE_TYPE_AOA: return "AOA";

case BT_DF_CTE_TYPE_AOD_1US: return "AOD 1 [us]";

case BT_DF_CTE_TYPE_AOD_2US: return "AOD 2 [us]";

case BT_DF_CTE_TYPE_NONE: return "";

default: return "Unknown";

}

}

 

static const char *packet_status2str(uint8_t status)

{

switch (status) {

case BT_DF_CTE_CRC_OK: return "CRC OK";

case BT_DF_CTE_CRC_ERR_CTE_BASED_TIME: return "CRC not OK, CTE Info OK";

case BT_DF_CTE_CRC_ERR_CTE_BASED_OTHER: return "CRC not OK, Sampled other way";

case BT_DF_CTE_INSUFFICIENT_RESOURCES: return "No resources";

default: return "Unknown";

}

}

 

static bool data_cb(struct bt_data *data, void *user_data)

{

char *name = user_data;

uint8_t len;

 

switch (data->type) {

case BT_DATA_NAME_SHORTENED:

case BT_DATA_NAME_COMPLETE:

len = MIN(data->data_len, PEER_NAME_LEN_MAX - 1);

memcpy(name, data->data, len);

name[len] = '\0';

return false;

default:

return true;

}

}

 

static void sync_cb(struct bt_le_per_adv_sync *sync,

    struct bt_le_per_adv_sync_synced_info *info)

{

char le_addr[BT_ADDR_LE_STR_LEN];

 

bt_addr_le_to_str(info->addr, le_addr, sizeof(le_addr));

 

printk("PER_ADV_SYNC[%u]: [DEVICE]: %s synced, "

       "Interval 0x%04x (%u ms), PHY %s\n",

       bt_le_per_adv_sync_get_index(sync), le_addr, info->interval,

       adv_interval_to_ms(info->interval), phy2str(info->phy));

 

k_sem_give(&sem_per_sync);

}

 

static void term_cb(struct bt_le_per_adv_sync *sync,

    const struct bt_le_per_adv_sync_term_info *info)

{

char le_addr[BT_ADDR_LE_STR_LEN];

 

bt_addr_le_to_str(info->addr, le_addr, sizeof(le_addr));

 

printk("PER_ADV_SYNC[%u]: [DEVICE]: %s sync terminated\n",

       bt_le_per_adv_sync_get_index(sync), le_addr);

 

k_sem_give(&sem_per_sync_lost);

}

 

static void recv_cb(struct bt_le_per_adv_sync *sync,

    const struct bt_le_per_adv_sync_recv_info *info,

    struct net_buf_simple *buf)

{

static char data_str[ADV_DATA_HEX_STR_LEN_MAX];

char le_addr[BT_ADDR_LE_STR_LEN];

 

bt_addr_le_to_str(info->addr, le_addr, sizeof(le_addr));

bin2hex(buf->data, buf->len, data_str, sizeof(data_str));

 

printk("PER_ADV_SYNC[%u]: [DEVICE]: %s, tx_power %i, "

       "RSSI %i, CTE %s, data length %u, data: %s\n",

       bt_le_per_adv_sync_get_index(sync), le_addr, info->tx_power,

       info->rssi, cte_type2str(info->cte_type), buf->len, data_str);

}

 

static void cte_recv_cb(struct bt_le_per_adv_sync *sync,

struct bt_df_per_adv_sync_iq_samples_report const *report)

{

printk("CTE[%u]: samples count %d, cte type %s, slot durations: %u [us], "

       "packet status %s, RSSI %i\n",

       bt_le_per_adv_sync_get_index(sync), report->sample_count,

       cte_type2str(report->cte_type), report->slot_durations,

       packet_status2str(report->packet_status), report->rssi);

printk("Channel index:%x  Id of antenna used:%x  Value of the paEventCounter:%x  Type of IQ samples:%x\n", 

report->chan_idx, report->rssi_ant_id, report->per_evt_counter, report->sample_type);

uint8_t *i_q;

char buf[512];

i_q = report->sample;

for (uint8_t scnt = 0; scnt < report->sample_count; scnt++) {

// printk("I:%d   Q:%d   Count:%d\n", *(i_q + scnt*2) , *(i_q + scnt*2+1), scnt);

snprintk(buf+scnt*9, 10 ,"(%03d,%03d)",*(i_q + scnt*2), *(i_q + scnt*2+1));

}

printk("%s\n", buf);

}

 

static struct bt_le_per_adv_sync_cb sync_callbacks = {

.synced = sync_cb,

.term = term_cb,

.recv = recv_cb,

.cte_report_cb = cte_recv_cb,

};

 

static void scan_recv(const struct bt_le_scan_recv_info *info,

      struct net_buf_simple *buf)

{

char le_addr[BT_ADDR_LE_STR_LEN];

char name[PEER_NAME_LEN_MAX];

 

(void)memset(name, 0, sizeof(name));

 

bt_data_parse(buf, data_cb, name);

 

bt_addr_le_to_str(info->addr, le_addr, sizeof(le_addr));

 

printk("[DEVICE]: %s, AD evt type %u, Tx Pwr: %i, RSSI %i %s C:%u S:%u "

       "D:%u SR:%u E:%u Prim: %s, Secn: %s, Interval: 0x%04x (%u ms), "

       "SID: %u\n",

       le_addr, info->adv_type, info->tx_power, info->rssi, name,

       (info->adv_props & BT_GAP_ADV_PROP_CONNECTABLE) != 0,

       (info->adv_props & BT_GAP_ADV_PROP_SCANNABLE) != 0,

       (info->adv_props & BT_GAP_ADV_PROP_DIRECTED) != 0,

       (info->adv_props & BT_GAP_ADV_PROP_SCAN_RESPONSE) != 0,

       (info->adv_props & BT_GAP_ADV_PROP_EXT_ADV) != 0, phy2str(info->primary_phy),

       phy2str(info->secondary_phy), info->interval, adv_interval_to_ms(info->interval),

       info->sid);

 

if (!per_adv_found && info->interval) {

for(uint8_t i=0; i<device_no; i++)

{  

printk("%s\n", name_buf[i]);

if(strcmp(name_buf[i], name) == 0){

printk("detect device : %d\n", device_no);

per_adv_found = true;

}

}

if(device_no == MAX_BEACON_DEVICE)

{

printk("max device : %d\n", device_no);

per_adv_found = true;

}

if(per_adv_found == false)

{

strcpy(name_buf[device_no], name);

sync_create_timeout_ms[device_no] =

adv_interval_to_ms(info->interval) * SYNC_CREATE_TIMEOUT_INTERVAL_NUM;

per_sid[device_no] = info->sid;

bt_addr_le_copy(&per_addr[device_no], info->addr);

device_no++;

printk("device_no : %d\n", device_no);

}

// sync_create_timeout_ms =

// adv_interval_to_ms(info->interval) * SYNC_CREATE_TIMEOUT_INTERVAL_NUM;

// per_adv_found = true;

// per_sid = info->sid;

// bt_addr_le_copy(&per_addr, info->addr);

if(per_adv_found == true)

{

k_sem_give(&sem_per_adv);

for(uint8_t i=device_no; i<MAX_BEACON_DEVICE; i++)

{

strcpy(name_buf[device_no], " ");

}

 

}

}

}

 

static struct bt_le_scan_cb scan_callbacks = {

.recv = scan_recv,

};

 

static void create_sync(uint8_t device)

{

struct bt_le_per_adv_sync_param sync_create_param;

int err;

 

printk("Creating Periodic Advertising Sync...");

bt_addr_le_copy(&sync_create_param.addr, &per_addr[device]);

sync_create_param.options = 0;

sync_create_param.sid = per_sid[device];

sync_create_param.skip = 0;

sync_create_param.timeout = 0xa;

err = bt_le_per_adv_sync_create(&sync_create_param, &sync[device]);

if (err) {

printk("failed (err %d)\n", err);

return;

}

printk("success.\n");

}

 

static int delete_sync(uint8_t device)

{

int err;

 

printk("Deleting Periodic Advertising Sync...");

err = bt_le_per_adv_sync_delete(sync[device]);

if (err) {

printk("failed (err %d)\n", err);

return err;

}

printk("success\n");

return 0;

}

 

static void enable_cte_rx(uint8_t device)

{

int err;

 

const struct bt_df_per_adv_sync_cte_rx_param cte_rx_params = {

.max_cte_count = 5,

#if defined(CONFIG_BT_DF_CTE_RX_AOA)

.cte_types = BT_DF_CTE_TYPE_ALL,

.slot_durations = 0x2,

.num_ant_ids = ARRAY_SIZE(ant_patterns),

.ant_ids = ant_patterns,

#else

.cte_types = BT_DF_CTE_TYPE_AOD_1US | BT_DF_CTE_TYPE_AOD_2US,

#endif /* CONFIG_BT_DF_CTE_RX_AOA */

};

 

printk("Enable receiving of CTE...\n");

err = bt_df_per_adv_sync_cte_rx_enable(sync[device], &cte_rx_params);

if (err) {

printk("failed (err %d)\n", err);

return;

}

printk("success. CTE receive enabled.\n");

}

 

static int scan_init(void)

{

printk("Scan callbacks register...");

bt_le_scan_cb_register(&scan_callbacks);

printk("success.\n");

 

printk("Periodic Advertising callbacks register...");

bt_le_per_adv_sync_cb_register(&sync_callbacks);

printk("success.\n");

 

return 0;

}

 

static int scan_enable(void)

{

struct bt_le_scan_param param = {

.type = BT_LE_SCAN_TYPE_ACTIVE,

// .options = BT_LE_SCAN_OPT_FILTER_DUPLICATE,

.options = BT_LE_SCAN_OPT_NONE,

.interval = BT_GAP_SCAN_FAST_INTERVAL,

.window = BT_GAP_SCAN_FAST_WINDOW,

.timeout = 0U,

};

 

int err;

 

if (!scan_enabled) {

printk("Start scanning...");

err = bt_le_scan_start(&param, NULL);

if (err) {

printk("failed (err %d)\n", err);

return err;

}

printk("success\n");

scan_enabled = true;

}

 

return 0;

}

 

static void scan_disable(void)

{

int err;

 

printk("Scan disable...");

err = bt_le_scan_stop();

if (err) {

printk("failed (err %d)\n", err);

return;

}

printk("Success.\n");

 

scan_enabled = false;

}

 

int main(void)

{

int err;

 

printk("Starting Connectionless Locator Demo\n");

 

printk("Bluetooth initialization...");

err = bt_enable(NULL);

if (err) {

printk("failed (err %d)\n", err);

}

printk("success\n");

 

scan_init();

 

scan_enabled = false;

 

while (true) {

per_adv_found = false;

scan_enable();

 

printk("Waiting for periodic advertising...\n");

err = k_sem_take(&sem_per_adv, K_FOREVER);

if (err) {

printk("failed (err %d)\n", err);

return 0;

}

printk("success. Found periodic advertising.\n");

 

/* Disable scan to cleanup output */

scan_disable();

 

for(uint8_t i=0; i<device_no; i++)

{

create_sync(i);

 

printk("Waiting for periodic sync...\n");

err = k_sem_take(&sem_per_sync, K_MSEC(sync_create_timeout_ms[i]));

if (err) {

printk("failed (err %d)\n", err);

err = delete_sync(i);

if (err) {

return 0;

}

continue;

}

printk("success. Periodic sync established.\n");

 

enable_cte_rx(i);

}

 

 

printk("Waiting for periodic sync lost...\n");

err = k_sem_take(&sem_per_sync_lost, K_FOREVER);

if (err) {

printk("failed (err %d)\n", err);

return 0;

}

printk("Periodic sync lost.\n");

}

}

 

 


###### prj.conf ######

 

#

# Copyright (c) 2021 Nordic Semiconductor ASA

#

# SPDX-License-Identifier: LicenseRef-Nordic-5-Clause

#

 

CONFIG_BT=y

CONFIG_BT_DEVICE_NAME="DF Connectionless Locator App"

 

CONFIG_BT_EXT_ADV=y

CONFIG_BT_PER_ADV_SYNC=y

CONFIG_BT_OBSERVER=y

 

# Enable Direction Finding Feature including AoA and AoD

CONFIG_BT_DF=y

CONFIG_BT_DF_CONNECTIONLESS_CTE_RX=y

 

CONFIG_BT_PER_ADV_SYNC_MAX=7

 

CONFIG_USE_SEGGER_RTT=y

CONFIG_RTT_CONSOLE=y

CONFIG_UART_CONSOLE=n

CONFIG_LOG_BACKEND_RTT=y

CONFIG_LOG_BACKEND_UART=n