2018-10-19 13:59:28 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2019-08-29 10:04:08 +02:00
|
|
|
* @ingroup pkg_nimble
|
2018-10-19 13:59:28 +02:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Glue code for running NimBLE for RIOT
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2020-10-21 15:58:21 +02:00
|
|
|
#include <assert.h>
|
|
|
|
|
2018-10-19 13:59:28 +02:00
|
|
|
#include "thread.h"
|
|
|
|
#include "nimble_riot.h"
|
|
|
|
|
|
|
|
#include "nimble/nimble_port.h"
|
2019-03-07 16:20:36 +01:00
|
|
|
#include "host/ble_hs.h"
|
|
|
|
#include "host/util/util.h"
|
2018-10-19 13:59:28 +02:00
|
|
|
|
2019-03-21 21:39:15 +01:00
|
|
|
#ifdef MODULE_NIMBLE_SVC_GAP
|
2018-10-19 13:59:28 +02:00
|
|
|
#include "services/gap/ble_svc_gap.h"
|
2019-03-21 21:39:15 +01:00
|
|
|
#endif
|
|
|
|
#ifdef MODULE_NIMBLE_SVC_GATT
|
2018-10-19 13:59:28 +02:00
|
|
|
#include "services/gatt/ble_svc_gatt.h"
|
2019-03-21 21:39:15 +01:00
|
|
|
#endif
|
2019-03-27 11:29:10 +01:00
|
|
|
#ifdef MODULE_NIMBLE_SVC_IPSS
|
|
|
|
#include "services/ipss/ble_svc_ipss.h"
|
|
|
|
#endif
|
2018-10-19 13:59:28 +02:00
|
|
|
|
2020-03-02 17:40:26 +01:00
|
|
|
#ifdef MODULE_NIMBLE_STATCONN
|
|
|
|
#include "nimble_statconn.h"
|
|
|
|
#endif
|
|
|
|
|
2021-11-22 10:35:37 +01:00
|
|
|
#ifdef MODULE_NIMBLE_AUTOADV
|
|
|
|
#include "nimble_autoadv_params.h"
|
|
|
|
#include "nimble_autoadv.h"
|
|
|
|
#endif
|
|
|
|
|
2019-08-29 15:56:24 +02:00
|
|
|
#if defined(MODULE_NIMBLE_AUTOCONN) && !defined(MODULE_NIMBLE_AUTOCONN_NOAUTOINIT)
|
|
|
|
#include "nimble_autoconn.h"
|
|
|
|
#include "nimble_autoconn_params.h"
|
|
|
|
#endif
|
|
|
|
|
2019-02-22 13:55:16 +01:00
|
|
|
#ifdef MODULE_NIMBLE_CONTROLLER
|
2019-04-27 21:24:54 +02:00
|
|
|
#if defined(CPU_FAM_NRF52) || defined(CPU_FAM_NRF51)
|
2018-10-19 13:59:28 +02:00
|
|
|
#include "nrf_clock.h"
|
|
|
|
#endif
|
2021-04-15 12:50:09 +02:00
|
|
|
#include "controller/ble_ll.h"
|
2018-10-19 13:59:28 +02:00
|
|
|
|
2022-08-13 08:33:40 +02:00
|
|
|
static char _stack_controller[NIMBLE_CONTROLLER_STACKSIZE];
|
|
|
|
#endif
|
|
|
|
|
2021-04-20 14:45:30 +02:00
|
|
|
#ifdef MODULE_NIMBLE_RPBLE
|
|
|
|
#include "nimble_rpble.h"
|
|
|
|
#include "nimble_rpble_params.h"
|
|
|
|
#endif
|
|
|
|
|
2019-02-22 13:55:16 +01:00
|
|
|
#ifdef MODULE_NIMBLE_HOST
|
2018-10-19 13:59:28 +02:00
|
|
|
static char _stack_host[NIMBLE_HOST_STACKSIZE];
|
|
|
|
|
2019-03-07 16:20:36 +01:00
|
|
|
uint8_t nimble_riot_own_addr_type;
|
|
|
|
|
2018-10-19 13:59:28 +02:00
|
|
|
static void *_host_thread(void *arg)
|
|
|
|
{
|
|
|
|
(void)arg;
|
|
|
|
|
|
|
|
nimble_port_init();
|
|
|
|
|
2019-02-22 13:55:16 +01:00
|
|
|
#ifdef MODULE_NIMBLE_CONTROLLER
|
2018-10-19 13:59:28 +02:00
|
|
|
/* XXX: NimBLE needs the nRF5x's LF clock to run */
|
2019-04-27 21:24:54 +02:00
|
|
|
#if defined(CPU_FAM_NRF52) || defined(CPU_FAM_NRF51)
|
2018-10-19 13:59:28 +02:00
|
|
|
clock_start_lf();
|
|
|
|
#endif
|
2019-04-27 21:24:54 +02:00
|
|
|
|
2018-10-19 13:59:28 +02:00
|
|
|
/* Run the controller
|
|
|
|
*
|
|
|
|
* Create task where NimBLE LL will run. This one is required as LL has its
|
|
|
|
* own event queue and should have highest priority.
|
|
|
|
*/
|
|
|
|
thread_create(_stack_controller, sizeof(_stack_controller),
|
|
|
|
NIMBLE_CONTROLLER_PRIO,
|
2024-03-07 15:51:39 +01:00
|
|
|
0,
|
2018-10-19 13:59:28 +02:00
|
|
|
(thread_task_func_t)nimble_port_ll_task_func, NULL,
|
|
|
|
"nimble_ctrl");
|
2021-04-15 12:50:09 +02:00
|
|
|
|
|
|
|
/* XXX: seeding of the used PRNG is done when this function is called the
|
|
|
|
* first time. However, this could potentially be in interrupt context,
|
|
|
|
* leading to an malloc call in that context, breaking with the used
|
|
|
|
* thread safe malloc wrapper in RIOT. So we better do this seeding in
|
|
|
|
* a deterministic fashion right here.
|
|
|
|
* -> this fix is temporary until a proper fix is merged to NimBLE
|
|
|
|
* upstream */
|
|
|
|
ble_ll_rand();
|
2019-02-22 13:55:16 +01:00
|
|
|
#endif
|
2018-10-19 13:59:28 +02:00
|
|
|
|
2019-03-04 17:39:38 +01:00
|
|
|
nimble_port_run();
|
|
|
|
|
|
|
|
/* never reached */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void nimble_riot_init(void)
|
|
|
|
{
|
2019-08-29 15:56:24 +02:00
|
|
|
int res;
|
|
|
|
(void)res;
|
|
|
|
|
2021-07-07 11:27:54 +02:00
|
|
|
#if !IS_USED(MODULE_MYNEWT_CORE) && IS_ACTIVE(NIMBLE_CFG_CONTROLLER)
|
|
|
|
/* in mynewt-nimble and uwb-core OS_CPUTIMER_TIMER_NUM == 5 is NRF_RTC0,
|
|
|
|
for nimble this must be used for the BLE stack and must go through
|
|
|
|
mynewt timer initialization for it to work properly. The RTC frequency
|
|
|
|
should be set to the highest possible value so, 32768Hz */
|
|
|
|
assert(MYNEWT_VAL_OS_CPUTIME_TIMER_NUM == 5);
|
|
|
|
assert(MYNEWT_VAL_OS_CPUTIME_FREQ == 32768);
|
|
|
|
int rc = hal_timer_init(MYNEWT_VAL_OS_CPUTIME_TIMER_NUM, NULL);
|
|
|
|
assert(rc == 0);
|
|
|
|
rc = hal_timer_config(MYNEWT_VAL_OS_CPUTIME_TIMER_NUM,
|
|
|
|
MYNEWT_VAL_OS_CPUTIME_FREQ);
|
|
|
|
assert(rc == 0);
|
|
|
|
(void)rc;
|
|
|
|
#endif
|
|
|
|
|
2018-10-19 13:59:28 +02:00
|
|
|
/* and finally initialize and run the host */
|
|
|
|
thread_create(_stack_host, sizeof(_stack_host),
|
|
|
|
NIMBLE_HOST_PRIO,
|
2024-03-07 15:51:39 +01:00
|
|
|
0,
|
2018-10-19 13:59:28 +02:00
|
|
|
_host_thread, NULL,
|
|
|
|
"nimble_host");
|
2019-03-07 16:19:23 +01:00
|
|
|
|
|
|
|
/* make sure synchronization of host and controller is done, this should
|
|
|
|
* always be the case at this point */
|
|
|
|
while (!ble_hs_synced()) {}
|
2019-03-07 16:20:36 +01:00
|
|
|
|
|
|
|
/* for reducing code duplication, we read our own address type once here
|
2023-10-16 12:17:48 +02:00
|
|
|
* so it can be reused later on */
|
2019-08-29 15:56:24 +02:00
|
|
|
res = ble_hs_util_ensure_addr(0);
|
2019-03-07 16:20:36 +01:00
|
|
|
assert(res == 0);
|
|
|
|
res = ble_hs_id_infer_auto(0, &nimble_riot_own_addr_type);
|
|
|
|
assert(res == 0);
|
2019-03-07 16:21:08 +01:00
|
|
|
|
2019-05-17 11:15:49 +02:00
|
|
|
#ifdef MODULE_NIMBLE_NETIF
|
|
|
|
extern void nimble_netif_init(void);
|
|
|
|
nimble_netif_init();
|
2022-09-21 09:36:17 +02:00
|
|
|
#ifdef MODULE_SHELL_CMD_NIMBLE_NETIF
|
2019-07-05 13:49:40 +02:00
|
|
|
extern void sc_nimble_netif_init(void);
|
|
|
|
sc_nimble_netif_init();
|
|
|
|
#endif
|
2019-05-17 11:15:49 +02:00
|
|
|
#endif
|
|
|
|
|
2019-10-23 21:18:08 +02:00
|
|
|
/* initialize the configured, built-in services */
|
2019-03-07 16:21:08 +01:00
|
|
|
#ifdef MODULE_NIMBLE_SVC_GAP
|
|
|
|
ble_svc_gap_init();
|
|
|
|
#endif
|
|
|
|
#ifdef MODULE_NIMBLE_SVC_GATT
|
|
|
|
ble_svc_gatt_init();
|
|
|
|
#endif
|
2019-03-27 11:29:10 +01:00
|
|
|
#ifdef MODULE_NIMBLE_SVC_IPSS
|
|
|
|
ble_svc_ipss_init();
|
|
|
|
#endif
|
2019-08-29 15:56:24 +02:00
|
|
|
|
2020-03-02 17:40:26 +01:00
|
|
|
#ifdef MODULE_NIMBLE_STATCONN
|
|
|
|
nimble_statconn_init();
|
|
|
|
#endif
|
|
|
|
|
2019-08-29 15:56:24 +02:00
|
|
|
#if defined(MODULE_NIMBLE_AUTOCONN) && !defined(MODULE_NIMBLE_AUTOCONN_NOAUTOINIT)
|
|
|
|
ble_gatts_start();
|
|
|
|
/* CAUTION: this must be called after nimble_netif_init() and also only
|
|
|
|
* after the GATT server has been initialized */
|
|
|
|
res = nimble_autoconn_init(&nimble_autoconn_params, NULL, 0);
|
|
|
|
assert(res == NIMBLE_AUTOCONN_OK);
|
|
|
|
nimble_autoconn_enable();
|
|
|
|
#endif
|
2020-02-24 11:38:17 +01:00
|
|
|
|
2019-06-13 14:25:45 +02:00
|
|
|
#ifdef MODULE_STDIO_NIMBLE
|
|
|
|
extern void stdio_nimble_init(void);
|
|
|
|
/* stdio_nimble_init() needs to be called after nimble stack initialization
|
|
|
|
* and before nimble_autoadv_init() */
|
|
|
|
stdio_nimble_init();
|
|
|
|
#endif
|
|
|
|
|
2020-02-24 11:38:17 +01:00
|
|
|
#ifdef MODULE_NIMBLE_AUTOADV
|
2021-11-22 10:35:37 +01:00
|
|
|
nimble_autoadv_init(&nimble_autoadv_cfg);
|
2020-02-24 11:38:17 +01:00
|
|
|
#endif
|
2021-04-20 14:45:30 +02:00
|
|
|
|
|
|
|
#ifdef MODULE_NIMBLE_RPBLE
|
|
|
|
res = nimble_rpble_init(&nimble_rpble_params);
|
|
|
|
assert(res == 0);
|
|
|
|
#endif
|
2018-10-19 13:59:28 +02:00
|
|
|
}
|
2022-03-28 09:16:54 +02:00
|
|
|
|
|
|
|
#if MYNEWT_VAL_BLE_EXT_ADV
|
|
|
|
int nimble_riot_get_phy_hci(uint8_t mode)
|
|
|
|
{
|
|
|
|
switch (mode) {
|
|
|
|
case NIMBLE_PHY_1M:
|
|
|
|
return BLE_HCI_LE_PHY_1M;
|
|
|
|
#if IS_USED(MODULE_NIMBLE_PHY_2MBIT)
|
|
|
|
case NIMBLE_PHY_2M:
|
|
|
|
return BLE_HCI_LE_PHY_2M;
|
|
|
|
#endif
|
|
|
|
#if IS_USED(MODULE_NIMBLE_PHY_CODED)
|
|
|
|
case NIMBLE_PHY_CODED:
|
|
|
|
return BLE_HCI_LE_PHY_CODED;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|