1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00

Merge pull request #12985 from haukepetersen/opt_nimble_autoconnusercb

pkg/nimble/autoconn: add user event callback
This commit is contained in:
Hauke Petersen 2019-12-18 17:50:22 +01:00 committed by GitHub
commit d9e9e4e288
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 100 additions and 44 deletions

View File

@ -111,6 +111,8 @@
#include <stdint.h> #include <stdint.h>
#include "nimble_netif.h"
#ifdef __cplusplus #ifdef __cplusplus
extern "C" { extern "C" {
#endif #endif
@ -168,6 +170,16 @@ typedef struct {
int nimble_autoconn_init(const nimble_autoconn_params_t *params, int nimble_autoconn_init(const nimble_autoconn_params_t *params,
const uint8_t *ad, size_t adlen); const uint8_t *ad, size_t adlen);
/**
* @brief Register a callback that is called on netif events
*
* The registered callback function is a simple pass-through of nimble_netif
* events. The callback is executed in the context of NimBLE's host thread.
*
* @param[in] cb event callback to register, may be NULL
*/
void nimble_autoconn_eventcb(nimble_netif_eventcb_t cb);
/** /**
* @brief Update the used parameters (timing and node ID) * @brief Update the used parameters (timing and node ID)
* *

View File

@ -65,6 +65,8 @@ static ble_npl_time_t _timeout_adv_period;
static ble_npl_time_t _timeout_scan_period; static ble_npl_time_t _timeout_scan_period;
static ble_npl_time_t _period_jitter; static ble_npl_time_t _period_jitter;
static nimble_netif_eventcb_t _eventcb = NULL;
/* this is run inside the NimBLE host thread */ /* this is run inside the NimBLE host thread */
static void _on_state_change(struct ble_npl_event *ev) static void _on_state_change(struct ble_npl_event *ev)
{ {
@ -140,41 +142,62 @@ static void _on_scan_evt(uint8_t type, const ble_addr_t *addr, int8_t rssi,
} }
} }
static void _on_netif_evt(int handle, nimble_netif_event_t event) static void _evt_dbg(const char *msg, int handle, const uint8_t *addr)
{ {
#if ENABLE_DEBUG
printf("%s (%i|", msg, handle);
bluetil_addr_print(addr);
puts(")");
#else
(void)msg;
(void)handle;
(void)addr;
#endif
}
static void _on_netif_evt(int handle, nimble_netif_event_t event,
const uint8_t *addr)
{
int en = 1;
switch (event) { switch (event) {
case NIMBLE_NETIF_CONNECTED_MASTER: case NIMBLE_NETIF_CONNECTED_MASTER:
DEBUG("[autoconn] CONNECTED as master %i\n", handle); _evt_dbg("[autoconn] CONNECTED master", handle, addr);
assert(_state == STATE_CONN); assert(_state == STATE_CONN);
_state = STATE_IDLE; _state = STATE_IDLE;
nimble_autoconn_enable();
break; break;
case NIMBLE_NETIF_CONNECTED_SLAVE: case NIMBLE_NETIF_CONNECTED_SLAVE:
DEBUG("[autoconn] CONNECTED as slave %i\n", handle); _evt_dbg("[autoconn] CONNECTED slave", handle, addr);
nimble_autoconn_enable();
break; break;
case NIMBLE_NETIF_CLOSED_MASTER: case NIMBLE_NETIF_CLOSED_MASTER:
DEBUG("[autoconn] CLOSED master connection\n"); _evt_dbg("[autoconn] CLOSED master", handle, addr);
nimble_autoconn_enable();
break; break;
case NIMBLE_NETIF_CLOSED_SLAVE: case NIMBLE_NETIF_CLOSED_SLAVE:
DEBUG("[autoconn] CLOSED slave connection\n"); _evt_dbg("[autoconn] CLOSED slave", handle, addr);
nimble_autoconn_enable();
break; break;
case NIMBLE_NETIF_CONNECT_ABORT: case NIMBLE_NETIF_CONNECT_ABORT:
DEBUG("[autoconn] CONNECT ABORT\n"); _evt_dbg("[autoconn] ABORTED", handle, addr);
assert(_state == STATE_CONN); assert(_state == STATE_CONN);
_state = STATE_IDLE; _state = STATE_IDLE;
nimble_autoconn_enable();
break; break;
case NIMBLE_NETIF_CONN_UPDATED: case NIMBLE_NETIF_CONN_UPDATED:
DEBUG("[autoconn] CONNECTION UPDATED %i\n", handle); _evt_dbg("[autoconn] UPDATED", handle, addr);
/* nothing to do here */ en = 0;
break; break;
default: default:
/* this should never happen */ /* this should never happen */
assert(0); assert(0);
} }
/* pass events to high-level user if someone registered for them */
if (_eventcb) {
_eventcb(handle, event, addr);
}
/* search for the next connection possibility */
if (en) {
nimble_autoconn_enable();
}
} }
static int _conn_update(nimble_netif_conn_t *conn, int handle, void *arg) static int _conn_update(nimble_netif_conn_t *conn, int handle, void *arg)
@ -196,6 +219,11 @@ int nimble_autoconn_init(const nimble_autoconn_params_t *params,
return nimble_autoconn_update(params, ad, adlen); return nimble_autoconn_update(params, ad, adlen);
} }
void nimble_autoconn_eventcb(nimble_netif_eventcb_t cb)
{
_eventcb = cb;
}
int nimble_autoconn_update(const nimble_autoconn_params_t *params, int nimble_autoconn_update(const nimble_autoconn_params_t *params,
const uint8_t *ad, size_t adlen) const uint8_t *ad, size_t adlen)
{ {
@ -282,8 +310,8 @@ int nimble_autoconn_update(const nimble_autoconn_params_t *params,
void nimble_autoconn_enable(void) void nimble_autoconn_enable(void)
{ {
DEBUG("[autoconn] ACTIVE\n");
if (nimble_netif_conn_count(NIMBLE_NETIF_UNUSED) > 0) { if (nimble_netif_conn_count(NIMBLE_NETIF_UNUSED) > 0) {
DEBUG("[autoconn] ACTIVE\n");
/* insert a random delay */ /* insert a random delay */
ble_npl_time_t delay = (ble_npl_time_t)random_uint32_range(0, ble_npl_time_t delay = (ble_npl_time_t)random_uint32_range(0,
(uint32_t)_period_jitter); (uint32_t)_period_jitter);
@ -294,8 +322,8 @@ void nimble_autoconn_enable(void)
void nimble_autoconn_disable(void) void nimble_autoconn_disable(void)
{ {
DEBUG("[autoconn] DISABLED\n");
if ((_state == STATE_ADV) || (_state == STATE_SCAN)) { if ((_state == STATE_ADV) || (_state == STATE_SCAN)) {
DEBUG("[autoconn] DISABLED\n");
_state = STATE_IDLE; _state = STATE_IDLE;
ble_npl_callout_stop(&_state_evt); ble_npl_callout_stop(&_state_evt);
nimble_scanner_stop(); nimble_scanner_stop();

View File

@ -150,8 +150,11 @@ enum {
* *
* @param[in] handle handle to the connection that triggered the event * @param[in] handle handle to the connection that triggered the event
* @param[in] event type of the event * @param[in] event type of the event
* @param[in] addr BLE address of the peer in the effected connection
*/ */
typedef void(*nimble_netif_eventcb_t)(int handle, nimble_netif_event_t event); typedef void(*nimble_netif_eventcb_t)(int handle,
nimble_netif_event_t event,
const uint8_t *addr);
/** /**
* @brief Initialize the netif implementation, spawns the netif thread * @brief Initialize the netif implementation, spawns the netif thread

View File

@ -159,7 +159,7 @@ int nimble_netif_conn_start_adv(void);
/** /**
* @brief Free the connection context with the given handle * @brief Free the connection context with the given handle
*/ */
void nimble_netif_conn_free(int handle); void nimble_netif_conn_free(int handle, uint8_t *addr);
/** /**
* @brief Find the connection context with a given GAP handle and return a * @brief Find the connection context with a given GAP handle and return a

View File

@ -79,10 +79,10 @@ static struct os_mempool _mem_pool;
static struct os_mbuf_pool _mbuf_pool; static struct os_mbuf_pool _mbuf_pool;
/* notify the user about state changes for a connection context */ /* notify the user about state changes for a connection context */
static void _notify(int handle, nimble_netif_event_t event) static void _notify(int handle, nimble_netif_event_t event, uint8_t *addr)
{ {
if (_eventcb) { if (_eventcb) {
_eventcb(handle, event); _eventcb(handle, event, addr);
} }
} }
@ -336,7 +336,7 @@ static int _on_l2cap_client_evt(struct ble_l2cap_event *event, void *arg)
conn->coc = event->connect.chan; conn->coc = event->connect.chan;
conn->state |= NIMBLE_NETIF_L2CAP_CLIENT; conn->state |= NIMBLE_NETIF_L2CAP_CLIENT;
conn->state &= ~NIMBLE_NETIF_CONNECTING; conn->state &= ~NIMBLE_NETIF_CONNECTING;
_notify(handle, NIMBLE_NETIF_CONNECTED_MASTER); _notify(handle, NIMBLE_NETIF_CONNECTED_MASTER, conn->addr);
break; break;
case BLE_L2CAP_EVENT_COC_DISCONNECTED: case BLE_L2CAP_EVENT_COC_DISCONNECTED:
assert(conn->coc); assert(conn->coc);
@ -374,7 +374,7 @@ static int _on_l2cap_server_evt(struct ble_l2cap_event *event, void *arg)
conn->coc = event->connect.chan; conn->coc = event->connect.chan;
conn->state |= NIMBLE_NETIF_L2CAP_SERVER; conn->state |= NIMBLE_NETIF_L2CAP_SERVER;
conn->state &= ~(NIMBLE_NETIF_ADV | NIMBLE_NETIF_CONNECTING); conn->state &= ~(NIMBLE_NETIF_ADV | NIMBLE_NETIF_CONNECTING);
_notify(handle, NIMBLE_NETIF_CONNECTED_SLAVE); _notify(handle, NIMBLE_NETIF_CONNECTED_SLAVE, conn->addr);
break; break;
case BLE_L2CAP_EVENT_COC_DISCONNECTED: case BLE_L2CAP_EVENT_COC_DISCONNECTED:
conn = nimble_netif_conn_from_gaphandle(event->disconnect.conn_handle); conn = nimble_netif_conn_from_gaphandle(event->disconnect.conn_handle);
@ -425,8 +425,9 @@ static int _on_gap_master_evt(struct ble_gap_event *event, void *arg)
switch (event->type) { switch (event->type) {
case BLE_GAP_EVENT_CONNECT: { case BLE_GAP_EVENT_CONNECT: {
if (event->connect.status != 0) { if (event->connect.status != 0) {
nimble_netif_conn_free(handle); uint8_t addr[BLE_ADDR_LEN];
_notify(handle, NIMBLE_NETIF_CONNECT_ABORT); nimble_netif_conn_free(handle, addr);
_notify(handle, NIMBLE_NETIF_CONNECT_ABORT, addr);
return 0; return 0;
} }
_on_gap_connected(conn, event->connect.conn_handle); _on_gap_connected(conn, event->connect.conn_handle);
@ -442,12 +443,14 @@ static int _on_gap_master_evt(struct ble_gap_event *event, void *arg)
assert(res == 0); assert(res == 0);
break; break;
} }
case BLE_GAP_EVENT_DISCONNECT: case BLE_GAP_EVENT_DISCONNECT: {
nimble_netif_conn_free(handle); uint8_t addr[BLE_ADDR_LEN];
_notify(handle, NIMBLE_NETIF_CLOSED_MASTER); nimble_netif_conn_free(handle, addr);
_notify(handle, NIMBLE_NETIF_CLOSED_MASTER, addr);
break; break;
}
case BLE_GAP_EVENT_CONN_UPDATE: case BLE_GAP_EVENT_CONN_UPDATE:
_notify(handle, NIMBLE_NETIF_CONN_UPDATED); _notify(handle, NIMBLE_NETIF_CONN_UPDATED, conn->addr);
break; break;
case BLE_GAP_EVENT_CONN_UPDATE_REQ: case BLE_GAP_EVENT_CONN_UPDATE_REQ:
case BLE_GAP_EVENT_MTU: case BLE_GAP_EVENT_MTU:
@ -469,8 +472,9 @@ static int _on_gap_slave_evt(struct ble_gap_event *event, void *arg)
switch (event->type) { switch (event->type) {
case BLE_GAP_EVENT_CONNECT: { case BLE_GAP_EVENT_CONNECT: {
if (event->connect.status != 0) { if (event->connect.status != 0) {
nimble_netif_conn_free(handle); uint8_t addr[BLE_ADDR_LEN];
_notify(handle, NIMBLE_NETIF_CONNECT_ABORT); nimble_netif_conn_free(handle, addr);
_notify(handle, NIMBLE_NETIF_CONNECT_ABORT, addr);
break; break;
} }
_on_gap_connected(conn, event->connect.conn_handle); _on_gap_connected(conn, event->connect.conn_handle);
@ -478,12 +482,14 @@ static int _on_gap_slave_evt(struct ble_gap_event *event, void *arg)
conn->state = NIMBLE_NETIF_GAP_SLAVE; conn->state = NIMBLE_NETIF_GAP_SLAVE;
break; break;
} }
case BLE_GAP_EVENT_DISCONNECT: case BLE_GAP_EVENT_DISCONNECT: {
nimble_netif_conn_free(handle); uint8_t addr[BLE_ADDR_LEN];
_notify(handle, NIMBLE_NETIF_CLOSED_SLAVE); nimble_netif_conn_free(handle, addr);
_notify(handle, NIMBLE_NETIF_CLOSED_SLAVE, addr);
break; break;
}
case BLE_GAP_EVENT_CONN_UPDATE: case BLE_GAP_EVENT_CONN_UPDATE:
_notify(handle, NIMBLE_NETIF_CONN_UPDATED); _notify(handle, NIMBLE_NETIF_CONN_UPDATED, conn->addr);
break; break;
case BLE_GAP_EVENT_CONN_UPDATE_REQ: case BLE_GAP_EVENT_CONN_UPDATE_REQ:
/* nothing to do here */ /* nothing to do here */
@ -609,7 +615,7 @@ int nimble_netif_accept_stop(void)
int res = ble_gap_adv_stop(); int res = ble_gap_adv_stop();
assert(res == 0); assert(res == 0);
(void)res; (void)res;
nimble_netif_conn_free(handle); nimble_netif_conn_free(handle, NULL);
return NIMBLE_NETIF_OK; return NIMBLE_NETIF_OK;
} }

View File

@ -156,12 +156,15 @@ int nimble_netif_conn_start_adv(void)
return handle; return handle;
} }
void nimble_netif_conn_free(int handle) void nimble_netif_conn_free(int handle, uint8_t *addr)
{ {
assert((handle >= 0) && (handle < CONN_CNT)); assert((handle >= 0) && (handle < CONN_CNT));
DEBUG("nimble_netif_conn_free, handle %i\n", handle); DEBUG("nimble_netif_conn_free, handle %i\n", handle);
mutex_lock(&_lock); mutex_lock(&_lock);
if (addr) {
memcpy(addr, _conn[handle].addr, BLE_ADDR_LEN);
}
memset(&_conn[handle], 0, sizeof(nimble_netif_conn_t)); memset(&_conn[handle], 0, sizeof(nimble_netif_conn_t));
_conn[handle].state = NIMBLE_NETIF_UNUSED; _conn[handle].state = NIMBLE_NETIF_UNUSED;
mutex_unlock(&_lock); mutex_unlock(&_lock);

View File

@ -58,13 +58,19 @@ static void _scan_for_name(uint8_t type, const ble_addr_t *addr, int8_t rssi,
} }
} }
static void _on_ble_evt(int handle, nimble_netif_event_t event) static void _print_evt(const char *msg, int handle, const uint8_t *addr)
{
printf("event: handle %i -> %s (", handle, msg);
bluetil_addr_print(addr);
puts(")");
}
static void _on_ble_evt(int handle, nimble_netif_event_t event,
const uint8_t *addr)
{ {
switch (event) { switch (event) {
case NIMBLE_NETIF_CONNECTED_MASTER: { case NIMBLE_NETIF_CONNECTED_MASTER: {
printf("event: handle %i -> CONNECTED as MASTER (", handle); _print_evt("CONNECTED as MASTER", handle, addr);
bluetil_addr_print(nimble_netif_conn_get(handle)->addr);
puts(")");
if (_name_to_connect != NULL) { if (_name_to_connect != NULL) {
printf("connection to '%s' established\n", _name_to_connect); printf("connection to '%s' established\n", _name_to_connect);
_name_to_connect = NULL; _name_to_connect = NULL;
@ -72,16 +78,14 @@ static void _on_ble_evt(int handle, nimble_netif_event_t event)
break; break;
} }
case NIMBLE_NETIF_CONNECTED_SLAVE: case NIMBLE_NETIF_CONNECTED_SLAVE:
printf("event: handle %i -> CONNECTED as SLAVE (", handle); _print_evt("CONNECTED as SLAVE", handle, addr);
bluetil_addr_print(nimble_netif_conn_get(handle)->addr);
puts(")");
break; break;
case NIMBLE_NETIF_CLOSED_MASTER: case NIMBLE_NETIF_CLOSED_MASTER:
case NIMBLE_NETIF_CLOSED_SLAVE: case NIMBLE_NETIF_CLOSED_SLAVE:
printf("event: handle %i -> CONNECTION CLOSED\n", handle); _print_evt("CONNECTION CLOSED", handle, addr);
break; break;
case NIMBLE_NETIF_CONNECT_ABORT: case NIMBLE_NETIF_CONNECT_ABORT:
printf("event: handle %i -> CONNECTION ABORT\n", handle); _print_evt("CONNECTION ABORT", handle, addr);
break; break;
case NIMBLE_NETIF_CONN_UPDATED: case NIMBLE_NETIF_CONN_UPDATED:
default: default: