2019-05-17 11:15:49 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018-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 pkg_nimble_netif
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief GNRC netif wrapper for NimBLE
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <limits.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "assert.h"
|
|
|
|
#include "thread.h"
|
|
|
|
#include "thread_flags.h"
|
|
|
|
|
|
|
|
#include "net/ble.h"
|
|
|
|
#include "net/bluetil/addr.h"
|
|
|
|
#include "net/gnrc/netif.h"
|
|
|
|
#include "net/gnrc/netif/hdr.h"
|
|
|
|
#include "net/gnrc/netreg.h"
|
|
|
|
#include "net/gnrc/pktbuf.h"
|
|
|
|
#include "net/gnrc/nettype.h"
|
|
|
|
|
|
|
|
#include "nimble_netif.h"
|
|
|
|
#include "nimble_netif_conn.h"
|
|
|
|
#include "nimble_riot.h"
|
|
|
|
#include "host/ble_gap.h"
|
|
|
|
#include "host/util/util.h"
|
2020-06-11 16:18:00 +02:00
|
|
|
#include "mem/mem.h"
|
2019-05-17 11:15:49 +02:00
|
|
|
|
2020-10-22 11:35:04 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2019-05-17 11:15:49 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
#ifdef MODULE_GNRC_SIXLOWPAN
|
|
|
|
#define NETTYPE GNRC_NETTYPE_SIXLOWPAN
|
|
|
|
#elif defined(MODULE_GNRC_IPV6)
|
|
|
|
#define NETTYPE GNRC_NETTYPE_IPV6
|
|
|
|
#else
|
|
|
|
#define NETTYPE GNRC_NETTYPE_UNDEF
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* thread flag used for signaling transmit readiness */
|
|
|
|
#define FLAG_TX_UNSTALLED (1u << 13)
|
2020-09-03 11:42:08 +02:00
|
|
|
#define FLAG_TX_NOTCONN (1u << 12)
|
|
|
|
#define FLAG_ALL (FLAG_TX_UNSTALLED | FLAG_TX_NOTCONN)
|
2019-05-17 11:15:49 +02:00
|
|
|
|
|
|
|
/* allocate a stack for the netif device */
|
|
|
|
static char _stack[THREAD_STACKSIZE_DEFAULT];
|
|
|
|
static thread_t *_netif_thread;
|
|
|
|
|
|
|
|
/* keep the actual device state */
|
2020-06-09 12:27:08 +02:00
|
|
|
static gnrc_netif_t _netif;
|
2019-05-17 11:15:49 +02:00
|
|
|
static gnrc_nettype_t _nettype = NETTYPE;
|
|
|
|
|
|
|
|
/* keep a reference to the event callback */
|
|
|
|
static nimble_netif_eventcb_t _eventcb;
|
|
|
|
|
|
|
|
/* notify the user about state changes for a connection context */
|
2019-12-18 16:17:16 +01:00
|
|
|
static void _notify(int handle, nimble_netif_event_t event, uint8_t *addr)
|
2019-05-17 11:15:49 +02:00
|
|
|
{
|
|
|
|
if (_eventcb) {
|
2019-12-18 16:17:16 +01:00
|
|
|
_eventcb(handle, event, addr);
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _netif_init(gnrc_netif_t *netif)
|
|
|
|
{
|
|
|
|
(void)netif;
|
|
|
|
|
2018-11-27 20:53:11 +01:00
|
|
|
gnrc_netif_default_init(netif);
|
2019-05-17 11:15:49 +02:00
|
|
|
/* save the threads context pointer, so we can set its flags */
|
2020-08-05 08:42:35 +02:00
|
|
|
_netif_thread = thread_get_active();
|
2019-05-17 11:15:49 +02:00
|
|
|
|
2020-05-05 17:55:11 +02:00
|
|
|
#if IS_USED(MODULE_GNRC_NETIF_6LO)
|
2019-05-17 11:15:49 +02:00
|
|
|
/* we disable fragmentation for this device, as the L2CAP layer takes care
|
|
|
|
* of this */
|
2020-06-09 12:27:08 +02:00
|
|
|
_netif.sixlo.max_frag_size = 0;
|
2020-05-05 17:55:11 +02:00
|
|
|
#endif /* IS_USED(MODULE_GNRC_NETIF_6LO) */
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int _send_pkt(nimble_netif_conn_t *conn, gnrc_pktsnip_t *pkt)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
int num_bytes = 0;
|
|
|
|
|
|
|
|
if (conn == NULL || conn->coc == NULL) {
|
|
|
|
return -ENOTCONN;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy the data into a newly allocated mbuf */
|
2020-11-16 18:38:15 +01:00
|
|
|
struct os_mbuf *sdu = os_msys_get_pkthdr(gnrc_pkt_len(pkt), 0);
|
2019-05-17 11:15:49 +02:00
|
|
|
if (sdu == NULL) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
while (pkt) {
|
|
|
|
res = os_mbuf_append(sdu, pkt->data, pkt->size);
|
|
|
|
if (res != 0) {
|
|
|
|
os_mbuf_free_chain(sdu);
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
num_bytes += (int)pkt->size;
|
|
|
|
pkt = pkt->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* send packet via the given L2CAP COC */
|
|
|
|
do {
|
|
|
|
res = ble_l2cap_send(conn->coc, sdu);
|
|
|
|
if (res == BLE_HS_EBUSY) {
|
2020-09-03 11:42:08 +02:00
|
|
|
thread_flags_t state = thread_flags_wait_any(FLAG_ALL);
|
|
|
|
/* abort if the active connection was lost in the mean time */
|
|
|
|
if ((state & FLAG_TX_NOTCONN) && !nimble_netif_conn_is_open(conn)) {
|
|
|
|
return -ECONNRESET;
|
|
|
|
}
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
} while (res == BLE_HS_EBUSY);
|
|
|
|
|
|
|
|
if ((res != 0) && (res != BLE_HS_ESTALLED)) {
|
|
|
|
os_mbuf_free_chain(sdu);
|
2021-02-03 12:10:52 +01:00
|
|
|
return -ECANCELED;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return num_bytes;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _netif_send(gnrc_netif_t *netif, gnrc_pktsnip_t *pkt)
|
|
|
|
{
|
|
|
|
assert(pkt->type == GNRC_NETTYPE_NETIF);
|
|
|
|
|
|
|
|
(void)netif;
|
|
|
|
int res;
|
|
|
|
|
|
|
|
gnrc_netif_hdr_t *hdr = (gnrc_netif_hdr_t *)pkt->data;
|
|
|
|
/* if packet is bcast or mcast, we send it to every connected node */
|
|
|
|
if (hdr->flags &
|
|
|
|
(GNRC_NETIF_HDR_FLAGS_BROADCAST | GNRC_NETIF_HDR_FLAGS_MULTICAST)) {
|
2020-04-27 15:16:27 +02:00
|
|
|
int handle = nimble_netif_conn_get_next(NIMBLE_NETIF_CONN_INVALID,
|
|
|
|
NIMBLE_NETIF_L2CAP_CONNECTED);
|
|
|
|
while (handle != NIMBLE_NETIF_CONN_INVALID) {
|
|
|
|
res = _send_pkt(nimble_netif_conn_get(handle), pkt->next);
|
|
|
|
handle = nimble_netif_conn_get_next(handle, NIMBLE_NETIF_L2CAP_CONNECTED);
|
|
|
|
}
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
/* send unicast */
|
|
|
|
else {
|
|
|
|
int handle = nimble_netif_conn_get_by_addr(
|
|
|
|
gnrc_netif_hdr_get_dst_addr(hdr));
|
|
|
|
nimble_netif_conn_t *conn = nimble_netif_conn_get(handle);
|
|
|
|
res = _send_pkt(conn, pkt->next);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* release the packet in GNRC's packet buffer */
|
|
|
|
gnrc_pktbuf_release(pkt);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* not used, we pass incoming data to GNRC directly from the NimBLE thread */
|
|
|
|
static gnrc_pktsnip_t *_netif_recv(gnrc_netif_t *netif)
|
|
|
|
{
|
|
|
|
(void)netif;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const gnrc_netif_ops_t _nimble_netif_ops = {
|
|
|
|
.init = _netif_init,
|
|
|
|
.send = _netif_send,
|
|
|
|
.recv = _netif_recv,
|
|
|
|
.get = gnrc_netif_get_from_netdev,
|
|
|
|
.set = gnrc_netif_set_from_netdev,
|
|
|
|
.msg_handler = NULL,
|
|
|
|
};
|
|
|
|
|
|
|
|
static inline int _netdev_init(netdev_t *dev)
|
|
|
|
{
|
2020-06-09 12:27:08 +02:00
|
|
|
(void)dev;
|
2019-05-17 11:15:49 +02:00
|
|
|
|
|
|
|
/* get our own address from the controller */
|
|
|
|
uint8_t tmp[6];
|
|
|
|
int res = ble_hs_id_copy_addr(nimble_riot_own_addr_type, tmp, NULL);
|
|
|
|
assert(res == 0);
|
|
|
|
(void)res;
|
|
|
|
|
2020-06-09 12:27:08 +02:00
|
|
|
bluetil_addr_swapped_cp(tmp, _netif.l2addr);
|
2019-05-17 11:15:49 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int _netdev_get(netdev_t *dev, netopt_t opt,
|
|
|
|
void *value, size_t max_len)
|
|
|
|
{
|
|
|
|
(void)dev;
|
|
|
|
int res = -ENOTSUP;
|
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
case NETOPT_ADDRESS:
|
|
|
|
assert(max_len >= BLE_ADDR_LEN);
|
2020-06-09 12:27:08 +02:00
|
|
|
memcpy(value, _netif.l2addr, BLE_ADDR_LEN);
|
2019-05-17 11:15:49 +02:00
|
|
|
res = BLE_ADDR_LEN;
|
|
|
|
break;
|
|
|
|
case NETOPT_ADDR_LEN:
|
|
|
|
case NETOPT_SRC_LEN:
|
|
|
|
assert(max_len == sizeof(uint16_t));
|
|
|
|
*((uint16_t *)value) = BLE_ADDR_LEN;
|
|
|
|
res = sizeof(uint16_t);
|
|
|
|
break;
|
2019-10-18 17:53:00 +02:00
|
|
|
case NETOPT_MAX_PDU_SIZE:
|
2019-05-17 11:15:49 +02:00
|
|
|
assert(max_len >= sizeof(uint16_t));
|
2020-11-16 18:38:15 +01:00
|
|
|
*((uint16_t *)value) = NIMBLE_NETIF_MTU;
|
2019-05-17 11:15:49 +02:00
|
|
|
res = sizeof(uint16_t);
|
|
|
|
break;
|
|
|
|
case NETOPT_PROTO:
|
|
|
|
assert(max_len == sizeof(gnrc_nettype_t));
|
|
|
|
*((gnrc_nettype_t *)value) = _nettype;
|
|
|
|
res = sizeof(gnrc_nettype_t);
|
|
|
|
break;
|
|
|
|
case NETOPT_DEVICE_TYPE:
|
|
|
|
assert(max_len == sizeof(uint16_t));
|
|
|
|
*((uint16_t *)value) = NETDEV_TYPE_BLE;
|
|
|
|
res = sizeof(uint16_t);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int _netdev_set(netdev_t *dev, netopt_t opt,
|
|
|
|
const void *value, size_t val_len)
|
|
|
|
{
|
|
|
|
(void)dev;
|
|
|
|
int res = -ENOTSUP;
|
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
case NETOPT_PROTO:
|
|
|
|
assert(val_len == sizeof(_nettype));
|
|
|
|
memcpy(&_nettype, value, sizeof(_nettype));
|
|
|
|
res = sizeof(_nettype);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const netdev_driver_t _nimble_netdev_driver = {
|
|
|
|
.send = NULL,
|
|
|
|
.recv = NULL,
|
|
|
|
.init = _netdev_init,
|
|
|
|
.isr = NULL,
|
|
|
|
.get = _netdev_get,
|
|
|
|
.set = _netdev_set,
|
|
|
|
};
|
|
|
|
|
|
|
|
static netdev_t _nimble_netdev_dummy = {
|
|
|
|
.driver = &_nimble_netdev_driver,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void _on_data(nimble_netif_conn_t *conn, struct ble_l2cap_event *event)
|
|
|
|
{
|
|
|
|
struct os_mbuf *rxb = event->receive.sdu_rx;
|
|
|
|
size_t rx_len = (size_t)OS_MBUF_PKTLEN(rxb);
|
|
|
|
|
|
|
|
/* allocate netif header */
|
|
|
|
gnrc_pktsnip_t *if_snip = gnrc_netif_hdr_build(conn->addr, BLE_ADDR_LEN,
|
2020-06-09 12:27:08 +02:00
|
|
|
_netif.l2addr,
|
2019-05-17 11:15:49 +02:00
|
|
|
BLE_ADDR_LEN);
|
|
|
|
if (if_snip == NULL) {
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* we need to add the device PID to the netif header */
|
2020-06-09 12:27:08 +02:00
|
|
|
gnrc_netif_hdr_set_netif(if_snip->data, &_netif);
|
2019-05-17 11:15:49 +02:00
|
|
|
|
|
|
|
/* allocate space in the pktbuf to store the packet */
|
|
|
|
gnrc_pktsnip_t *payload = gnrc_pktbuf_add(if_snip, NULL, rx_len, _nettype);
|
|
|
|
if (payload == NULL) {
|
|
|
|
gnrc_pktbuf_release(if_snip);
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy payload from mbuf into pktbuffer */
|
|
|
|
int res = os_mbuf_copydata(rxb, 0, rx_len, payload->data);
|
|
|
|
if (res != 0) {
|
|
|
|
gnrc_pktbuf_release(payload);
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* finally dispatch the receive packet to GNRC */
|
|
|
|
if (!gnrc_netapi_dispatch_receive(payload->type, GNRC_NETREG_DEMUX_CTX_ALL,
|
|
|
|
payload)) {
|
|
|
|
gnrc_pktbuf_release(payload);
|
|
|
|
}
|
|
|
|
|
|
|
|
end:
|
|
|
|
/* free the mbuf and allocate a new one for receiving new data */
|
|
|
|
os_mbuf_free_chain(rxb);
|
2020-11-16 18:38:15 +01:00
|
|
|
rxb = os_msys_get_pkthdr(MYNEWT_VAL_BLE_L2CAP_COC_MPS, 0);
|
2019-05-17 11:15:49 +02:00
|
|
|
/* due to buffer provisioning, there should always be enough space */
|
|
|
|
assert(rxb != NULL);
|
|
|
|
ble_l2cap_recv_ready(event->receive.chan, rxb);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _on_l2cap_client_evt(struct ble_l2cap_event *event, void *arg)
|
|
|
|
{
|
|
|
|
int handle = (int)arg;
|
|
|
|
nimble_netif_conn_t *conn = nimble_netif_conn_get(handle);
|
|
|
|
assert(conn && (conn->state & NIMBLE_NETIF_GAP_MASTER));
|
|
|
|
|
|
|
|
switch (event->type) {
|
|
|
|
case BLE_L2CAP_EVENT_COC_CONNECTED:
|
2020-01-10 15:50:53 +01:00
|
|
|
if (event->connect.status != 0) {
|
|
|
|
/* in the unlikely event the L2CAP connection establishment
|
|
|
|
* fails, we close the GAP connection */
|
|
|
|
ble_gap_terminate(conn->gaphandle, BLE_ERR_REM_USER_CONN_TERM);
|
|
|
|
break;
|
|
|
|
}
|
2019-05-17 11:15:49 +02:00
|
|
|
conn->coc = event->connect.chan;
|
|
|
|
conn->state |= NIMBLE_NETIF_L2CAP_CLIENT;
|
|
|
|
conn->state &= ~NIMBLE_NETIF_CONNECTING;
|
2019-12-18 16:17:16 +01:00
|
|
|
_notify(handle, NIMBLE_NETIF_CONNECTED_MASTER, conn->addr);
|
2019-05-17 11:15:49 +02:00
|
|
|
break;
|
|
|
|
case BLE_L2CAP_EVENT_COC_DISCONNECTED:
|
2020-01-10 15:47:28 +01:00
|
|
|
assert(conn->state & NIMBLE_NETIF_L2CAP_CLIENT);
|
2019-05-17 11:15:49 +02:00
|
|
|
conn->state &= ~NIMBLE_NETIF_L2CAP_CONNECTED;
|
|
|
|
break;
|
|
|
|
case BLE_L2CAP_EVENT_COC_ACCEPT:
|
|
|
|
/* this event should never be triggered for the L2CAP client */
|
|
|
|
assert(0);
|
|
|
|
break;
|
|
|
|
case BLE_L2CAP_EVENT_COC_DATA_RECEIVED:
|
|
|
|
_on_data(conn, event);
|
|
|
|
break;
|
|
|
|
case BLE_L2CAP_EVENT_COC_TX_UNSTALLED:
|
|
|
|
thread_flags_set(_netif_thread, FLAG_TX_UNSTALLED);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _on_l2cap_server_evt(struct ble_l2cap_event *event, void *arg)
|
|
|
|
{
|
|
|
|
(void)arg;
|
|
|
|
int handle;
|
|
|
|
nimble_netif_conn_t *conn;
|
|
|
|
|
|
|
|
switch (event->type) {
|
|
|
|
case BLE_L2CAP_EVENT_COC_CONNECTED:
|
|
|
|
handle = nimble_netif_conn_get_by_gaphandle(event->connect.conn_handle);
|
|
|
|
conn = nimble_netif_conn_get(handle);
|
|
|
|
assert(conn);
|
2020-01-10 15:50:53 +01:00
|
|
|
|
2021-07-01 09:36:32 +02:00
|
|
|
/* in the unlikely event the L2CAP connection establishment fails,
|
|
|
|
* we close the GAP connection */
|
2020-01-10 15:50:53 +01:00
|
|
|
if (event->connect.status != 0) {
|
|
|
|
ble_gap_terminate(conn->gaphandle, BLE_ERR_REM_USER_CONN_TERM);
|
|
|
|
break;
|
|
|
|
}
|
2021-07-01 09:36:32 +02:00
|
|
|
|
|
|
|
/* we need to update the state to keep everything in sync */
|
2019-05-17 11:15:49 +02:00
|
|
|
conn->coc = event->connect.chan;
|
|
|
|
conn->state |= NIMBLE_NETIF_L2CAP_SERVER;
|
|
|
|
conn->state &= ~(NIMBLE_NETIF_ADV | NIMBLE_NETIF_CONNECTING);
|
2021-07-01 09:36:32 +02:00
|
|
|
|
|
|
|
/* in case conn itvl spacing is enabled, make sure that the conn
|
|
|
|
* itvl of the new connection is sufficiently spaced */
|
|
|
|
if ((NIMBLE_NETIF_CONN_ITVL_SPACING > 0) &&
|
|
|
|
nimble_netif_conn_itvl_used(conn->itvl, handle)) {
|
|
|
|
ble_gap_terminate(conn->gaphandle, BLE_ERR_REM_USER_CONN_TERM);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-12-18 16:17:16 +01:00
|
|
|
_notify(handle, NIMBLE_NETIF_CONNECTED_SLAVE, conn->addr);
|
2019-05-17 11:15:49 +02:00
|
|
|
break;
|
|
|
|
case BLE_L2CAP_EVENT_COC_DISCONNECTED:
|
|
|
|
conn = nimble_netif_conn_from_gaphandle(event->disconnect.conn_handle);
|
2020-01-10 15:47:28 +01:00
|
|
|
assert(conn && (conn->state & NIMBLE_NETIF_L2CAP_SERVER));
|
2021-07-01 09:36:32 +02:00
|
|
|
conn->coc = NULL;
|
2019-05-17 11:15:49 +02:00
|
|
|
conn->state &= ~NIMBLE_NETIF_L2CAP_CONNECTED;
|
|
|
|
break;
|
|
|
|
case BLE_L2CAP_EVENT_COC_ACCEPT: {
|
2020-11-16 18:38:15 +01:00
|
|
|
struct os_mbuf *sdu_rx = os_msys_get_pkthdr(
|
|
|
|
MYNEWT_VAL_BLE_L2CAP_COC_MPS, 0);
|
2019-05-17 11:15:49 +02:00
|
|
|
/* there should always be enough buffer space */
|
|
|
|
assert(sdu_rx != NULL);
|
|
|
|
ble_l2cap_recv_ready(event->accept.chan, sdu_rx);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case BLE_L2CAP_EVENT_COC_DATA_RECEIVED:
|
|
|
|
conn = nimble_netif_conn_from_gaphandle(event->receive.conn_handle);
|
|
|
|
assert(conn);
|
|
|
|
_on_data(conn, event);
|
|
|
|
break;
|
|
|
|
case BLE_L2CAP_EVENT_COC_TX_UNSTALLED:
|
|
|
|
thread_flags_set(_netif_thread, FLAG_TX_UNSTALLED);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _on_gap_connected(nimble_netif_conn_t *conn, uint16_t conn_handle)
|
|
|
|
{
|
|
|
|
struct ble_gap_conn_desc desc;
|
|
|
|
int res = ble_gap_conn_find(conn_handle, &desc);
|
|
|
|
assert(res == 0);
|
|
|
|
(void)res;
|
|
|
|
|
|
|
|
conn->gaphandle = conn_handle;
|
2021-07-01 09:36:32 +02:00
|
|
|
conn->itvl = desc.conn_itvl;
|
2019-05-17 11:15:49 +02:00
|
|
|
bluetil_addr_swapped_cp(desc.peer_id_addr.val, conn->addr);
|
|
|
|
}
|
|
|
|
|
2021-07-01 09:36:32 +02:00
|
|
|
static void _on_gap_param_update(int handle, nimble_netif_conn_t *conn)
|
|
|
|
{
|
|
|
|
struct ble_gap_conn_desc desc;
|
|
|
|
int res = ble_gap_conn_find(conn->gaphandle, &desc);
|
|
|
|
assert(res == 0) ;
|
|
|
|
(void)res;
|
|
|
|
|
|
|
|
conn->itvl = desc.conn_itvl;
|
|
|
|
_notify(handle, NIMBLE_NETIF_CONN_UPDATED, conn->addr);
|
|
|
|
}
|
|
|
|
|
2019-05-17 11:15:49 +02:00
|
|
|
static int _on_gap_master_evt(struct ble_gap_event *event, void *arg)
|
|
|
|
{
|
|
|
|
int res = 0;
|
|
|
|
int handle = (int)arg;
|
|
|
|
nimble_netif_conn_t *conn = nimble_netif_conn_get(handle);
|
|
|
|
assert(conn);
|
|
|
|
|
|
|
|
switch (event->type) {
|
|
|
|
case BLE_GAP_EVENT_CONNECT: {
|
|
|
|
if (event->connect.status != 0) {
|
2019-12-18 16:17:16 +01:00
|
|
|
uint8_t addr[BLE_ADDR_LEN];
|
|
|
|
nimble_netif_conn_free(handle, addr);
|
2020-01-10 15:47:28 +01:00
|
|
|
_notify(handle, NIMBLE_NETIF_ABORT_MASTER, addr);
|
2019-05-17 11:15:49 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
_on_gap_connected(conn, event->connect.conn_handle);
|
|
|
|
conn->state |= NIMBLE_NETIF_GAP_MASTER;
|
|
|
|
|
2020-11-16 18:38:15 +01:00
|
|
|
struct os_mbuf *sdu_rx = os_msys_get_pkthdr(
|
|
|
|
MYNEWT_VAL_BLE_L2CAP_COC_MPS, 0);
|
|
|
|
/* in the rare case that we run out of buffers, we close the new
|
|
|
|
* connection right away */
|
|
|
|
if (sdu_rx == NULL) {
|
|
|
|
ble_gap_terminate(handle, BLE_ERR_REM_USER_CONN_TERM);
|
|
|
|
break;
|
|
|
|
}
|
2019-05-17 11:15:49 +02:00
|
|
|
res = ble_l2cap_connect(event->connect.conn_handle,
|
2020-11-16 18:38:15 +01:00
|
|
|
NIMBLE_NETIF_CID, NIMBLE_NETIF_MTU, sdu_rx,
|
2019-05-17 11:15:49 +02:00
|
|
|
_on_l2cap_client_evt, (void *)handle);
|
2020-11-16 18:38:15 +01:00
|
|
|
/* should always be successful */
|
2019-05-17 11:15:49 +02:00
|
|
|
assert(res == 0);
|
|
|
|
break;
|
|
|
|
}
|
2019-12-18 16:17:16 +01:00
|
|
|
case BLE_GAP_EVENT_DISCONNECT: {
|
2020-01-10 15:47:28 +01:00
|
|
|
nimble_netif_event_t type;
|
|
|
|
type = (conn->coc != NULL) ? NIMBLE_NETIF_CLOSED_MASTER
|
|
|
|
: NIMBLE_NETIF_ABORT_MASTER;
|
2019-12-18 16:17:16 +01:00
|
|
|
uint8_t addr[BLE_ADDR_LEN];
|
|
|
|
nimble_netif_conn_free(handle, addr);
|
2020-09-03 11:42:08 +02:00
|
|
|
thread_flags_set(_netif_thread, FLAG_TX_NOTCONN);
|
2020-01-10 15:47:28 +01:00
|
|
|
_notify(handle, type, addr);
|
2019-05-17 11:15:49 +02:00
|
|
|
break;
|
2019-12-18 16:17:16 +01:00
|
|
|
}
|
2019-05-17 11:15:49 +02:00
|
|
|
case BLE_GAP_EVENT_CONN_UPDATE:
|
2021-07-01 09:36:32 +02:00
|
|
|
_on_gap_param_update(handle, conn);
|
2019-05-17 11:15:49 +02:00
|
|
|
break;
|
|
|
|
case BLE_GAP_EVENT_CONN_UPDATE_REQ:
|
|
|
|
case BLE_GAP_EVENT_MTU:
|
|
|
|
/* nothing to do here */
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _on_gap_slave_evt(struct ble_gap_event *event, void *arg)
|
|
|
|
{
|
|
|
|
int handle = (int)arg;
|
|
|
|
nimble_netif_conn_t *conn = nimble_netif_conn_get(handle);
|
|
|
|
assert(conn);
|
|
|
|
|
|
|
|
switch (event->type) {
|
|
|
|
case BLE_GAP_EVENT_CONNECT: {
|
|
|
|
if (event->connect.status != 0) {
|
2019-12-18 16:17:16 +01:00
|
|
|
uint8_t addr[BLE_ADDR_LEN];
|
|
|
|
nimble_netif_conn_free(handle, addr);
|
2020-01-10 15:47:28 +01:00
|
|
|
_notify(handle, NIMBLE_NETIF_ABORT_SLAVE, addr);
|
2019-05-17 11:15:49 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
_on_gap_connected(conn, event->connect.conn_handle);
|
|
|
|
assert(conn->state == NIMBLE_NETIF_ADV);
|
|
|
|
conn->state = NIMBLE_NETIF_GAP_SLAVE;
|
2020-01-10 15:47:28 +01:00
|
|
|
_notify(handle, NIMBLE_NETIF_INIT_SLAVE, conn->addr);
|
2019-05-17 11:15:49 +02:00
|
|
|
break;
|
|
|
|
}
|
2019-12-18 16:17:16 +01:00
|
|
|
case BLE_GAP_EVENT_DISCONNECT: {
|
2020-01-10 15:47:28 +01:00
|
|
|
nimble_netif_event_t type;
|
|
|
|
type = (conn->coc != NULL) ? NIMBLE_NETIF_CLOSED_SLAVE
|
|
|
|
: NIMBLE_NETIF_ABORT_SLAVE;
|
2019-12-18 16:17:16 +01:00
|
|
|
uint8_t addr[BLE_ADDR_LEN];
|
|
|
|
nimble_netif_conn_free(handle, addr);
|
2020-09-03 11:42:08 +02:00
|
|
|
thread_flags_set(_netif_thread, FLAG_TX_NOTCONN);
|
2020-01-10 15:47:28 +01:00
|
|
|
_notify(handle, type, addr);
|
2019-05-17 11:15:49 +02:00
|
|
|
break;
|
2019-12-18 16:17:16 +01:00
|
|
|
}
|
2019-05-17 11:15:49 +02:00
|
|
|
case BLE_GAP_EVENT_CONN_UPDATE:
|
2021-07-01 09:36:32 +02:00
|
|
|
_on_gap_param_update(handle, conn);
|
2019-05-17 11:15:49 +02:00
|
|
|
break;
|
|
|
|
case BLE_GAP_EVENT_CONN_UPDATE_REQ:
|
|
|
|
/* nothing to do here */
|
|
|
|
break;
|
2021-07-01 10:54:59 +02:00
|
|
|
case BLE_GAP_EVENT_ADV_COMPLETE: {
|
|
|
|
uint8_t addr[BLE_ADDR_LEN];
|
|
|
|
nimble_netif_conn_free(handle, addr);
|
|
|
|
_notify(handle, NIMBLE_NETIF_ACCEPT_STOP, addr);
|
|
|
|
}
|
2019-05-17 11:15:49 +02:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void nimble_netif_init(void)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
(void)res;
|
|
|
|
|
|
|
|
/* setup the connection context table */
|
|
|
|
nimble_netif_conn_init();
|
|
|
|
|
2020-11-16 18:38:15 +01:00
|
|
|
res = ble_l2cap_create_server(NIMBLE_NETIF_CID, NIMBLE_NETIF_MTU,
|
2019-05-17 11:15:49 +02:00
|
|
|
_on_l2cap_server_evt, NULL);
|
|
|
|
assert(res == 0);
|
|
|
|
(void)res;
|
|
|
|
|
2019-12-19 19:23:15 +01:00
|
|
|
gnrc_netif_create(&_netif, _stack, sizeof(_stack), GNRC_NETIF_PRIO,
|
2019-05-17 11:15:49 +02:00
|
|
|
"nimble_netif", &_nimble_netdev_dummy, &_nimble_netif_ops);
|
|
|
|
}
|
|
|
|
|
|
|
|
void nimble_netif_eventcb(nimble_netif_eventcb_t cb)
|
|
|
|
{
|
|
|
|
_eventcb = cb;
|
|
|
|
}
|
|
|
|
|
|
|
|
int nimble_netif_connect(const ble_addr_t *addr,
|
2021-04-21 20:33:41 +02:00
|
|
|
struct ble_gap_conn_params *conn_params,
|
2019-05-17 11:15:49 +02:00
|
|
|
uint32_t timeout)
|
|
|
|
{
|
|
|
|
assert(addr);
|
|
|
|
assert(_eventcb);
|
|
|
|
|
2021-04-21 20:33:41 +02:00
|
|
|
uint16_t itvl_min = 0;
|
|
|
|
uint16_t itvl_max = 0;
|
|
|
|
|
2019-05-17 11:15:49 +02:00
|
|
|
/* the netif_conn module expects addresses in network byte order */
|
|
|
|
uint8_t addrn[BLE_ADDR_LEN];
|
|
|
|
bluetil_addr_swapped_cp(addr->val, addrn);
|
|
|
|
|
|
|
|
/* check that there is no open connection with the given address */
|
|
|
|
if (nimble_netif_conn_connected(addrn) ||
|
|
|
|
nimble_netif_conn_connecting()) {
|
2021-09-10 21:07:33 +02:00
|
|
|
return -EBUSY;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* get empty connection context */
|
|
|
|
int handle = nimble_netif_conn_start_connection(addrn);
|
|
|
|
if (handle == NIMBLE_NETIF_CONN_INVALID) {
|
2021-09-10 21:07:33 +02:00
|
|
|
return -ENOMEM;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
|
2021-04-21 20:33:41 +02:00
|
|
|
if ((conn_params != NULL)
|
|
|
|
&& (conn_params->itvl_min != conn_params->itvl_max)) {
|
|
|
|
/* we need to save the min/max intervals in order to restore them
|
|
|
|
* later on */
|
|
|
|
itvl_min = conn_params->itvl_min;
|
|
|
|
itvl_max = conn_params->itvl_max;
|
|
|
|
|
|
|
|
uint16_t itvl = nimble_netif_conn_gen_itvl(itvl_min, itvl_max);
|
|
|
|
if (itvl == 0) {
|
2021-09-10 21:07:33 +02:00
|
|
|
return -ECANCELED;
|
2021-04-21 20:33:41 +02:00
|
|
|
}
|
|
|
|
conn_params->itvl_min = itvl;
|
|
|
|
conn_params->itvl_max = itvl;
|
|
|
|
}
|
|
|
|
|
2019-05-17 11:15:49 +02:00
|
|
|
int res = ble_gap_connect(nimble_riot_own_addr_type, addr, timeout,
|
|
|
|
conn_params, _on_gap_master_evt, (void *)handle);
|
|
|
|
assert(res == 0);
|
|
|
|
(void)res;
|
|
|
|
|
2021-04-21 20:33:41 +02:00
|
|
|
if (itvl_min != itvl_max) {
|
|
|
|
conn_params->itvl_min = itvl_min;
|
|
|
|
conn_params->itvl_max = itvl_max;
|
|
|
|
}
|
|
|
|
|
2020-01-10 15:47:28 +01:00
|
|
|
_notify(handle, NIMBLE_NETIF_INIT_MASTER, addrn);
|
|
|
|
|
2019-05-17 11:15:49 +02:00
|
|
|
return handle;
|
|
|
|
}
|
|
|
|
|
|
|
|
int nimble_netif_close(int handle)
|
|
|
|
{
|
|
|
|
nimble_netif_conn_t *conn = nimble_netif_conn_get(handle);
|
|
|
|
if (conn == NULL) {
|
2021-09-10 21:07:33 +02:00
|
|
|
return -EINVAL;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
else if (!(conn->state & NIMBLE_NETIF_L2CAP_CONNECTED)) {
|
2021-09-10 21:07:33 +02:00
|
|
|
return -ENOTCONN;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int res = ble_gap_terminate(ble_l2cap_get_conn_handle(conn->coc),
|
|
|
|
BLE_ERR_REM_USER_CONN_TERM);
|
|
|
|
assert(res == 0);
|
|
|
|
(void)res;
|
|
|
|
|
2021-09-10 21:07:33 +02:00
|
|
|
return 0;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
|
2021-07-01 10:54:59 +02:00
|
|
|
static int _accept(const uint8_t *ad, size_t ad_len, const ble_addr_t *addr,
|
|
|
|
uint32_t timeout,
|
|
|
|
const struct ble_gap_adv_params *adv_params)
|
2019-05-17 11:15:49 +02:00
|
|
|
{
|
|
|
|
assert(adv_params);
|
|
|
|
|
|
|
|
int handle;
|
|
|
|
int res;
|
|
|
|
(void)res;
|
|
|
|
|
|
|
|
/* allocate a connection context for incoming connections */
|
|
|
|
handle = nimble_netif_conn_start_adv();
|
|
|
|
if (handle < 0) {
|
|
|
|
return handle;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set advertisement data */
|
2021-07-01 10:54:59 +02:00
|
|
|
if (ad != NULL) {
|
|
|
|
res = ble_gap_adv_set_data(ad, (int)ad_len);
|
|
|
|
assert(res == 0);
|
|
|
|
}
|
|
|
|
/* remember address if applicable */
|
|
|
|
if (addr) {
|
|
|
|
nimble_netif_conn_t *conn = nimble_netif_conn_get(handle);
|
|
|
|
bluetil_addr_swapped_cp(addr->val, conn->addr);
|
|
|
|
}
|
|
|
|
|
2019-05-17 11:15:49 +02:00
|
|
|
/* remember context and start advertising */
|
2021-07-01 10:54:59 +02:00
|
|
|
res = ble_gap_adv_start(nimble_riot_own_addr_type, addr, timeout,
|
2019-05-17 11:15:49 +02:00
|
|
|
adv_params, _on_gap_slave_evt, (void *)handle);
|
|
|
|
assert(res == 0);
|
|
|
|
|
2020-06-09 12:27:08 +02:00
|
|
|
_notify(handle, NIMBLE_NETIF_ACCEPTING, _netif.l2addr);
|
2020-01-10 15:47:28 +01:00
|
|
|
|
2021-09-10 21:07:33 +02:00
|
|
|
return 0;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
|
2021-07-01 10:54:59 +02:00
|
|
|
int nimble_netif_accept(const uint8_t *ad, size_t ad_len,
|
|
|
|
const struct ble_gap_adv_params *adv_params)
|
|
|
|
{
|
|
|
|
assert(ad != NULL);
|
|
|
|
assert(ad_len > 0);
|
|
|
|
return _accept(ad, ad_len, NULL, BLE_HS_FOREVER, adv_params);
|
|
|
|
}
|
|
|
|
|
|
|
|
int nimble_netif_accept_direct(const ble_addr_t *addr, uint32_t timeout,
|
|
|
|
const struct ble_gap_adv_params *adv_params)
|
|
|
|
{
|
|
|
|
return _accept(NULL, 0, addr, timeout, adv_params);
|
|
|
|
}
|
|
|
|
|
2019-05-17 11:15:49 +02:00
|
|
|
int nimble_netif_accept_stop(void)
|
|
|
|
{
|
|
|
|
int handle = nimble_netif_conn_get_adv();
|
|
|
|
if (handle == NIMBLE_NETIF_CONN_INVALID) {
|
2021-09-10 21:07:33 +02:00
|
|
|
return -EALREADY;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int res = ble_gap_adv_stop();
|
|
|
|
assert(res == 0);
|
|
|
|
(void)res;
|
2019-12-18 16:17:16 +01:00
|
|
|
nimble_netif_conn_free(handle, NULL);
|
2020-06-09 12:27:08 +02:00
|
|
|
_notify(handle, NIMBLE_NETIF_ACCEPT_STOP, _netif.l2addr);
|
2019-05-17 11:15:49 +02:00
|
|
|
|
2021-09-10 21:07:33 +02:00
|
|
|
return 0;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int nimble_netif_update(int handle,
|
|
|
|
const struct ble_gap_upd_params *conn_params)
|
|
|
|
{
|
|
|
|
nimble_netif_conn_t *conn = nimble_netif_conn_get(handle);
|
|
|
|
if (conn == NULL) {
|
2021-09-10 21:07:33 +02:00
|
|
|
return -ENOTCONN;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int res = ble_gap_update_params(conn->gaphandle, conn_params);
|
|
|
|
if (res != 0) {
|
2021-09-10 21:07:33 +02:00
|
|
|
return -ECANCELED;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
|
|
|
|
2021-09-10 21:07:33 +02:00
|
|
|
return 0;
|
2019-05-17 11:15:49 +02:00
|
|
|
}
|
2021-02-03 12:11:26 +01:00
|
|
|
|
|
|
|
int nimble_netif_used_chanmap(int handle, uint8_t map[5])
|
|
|
|
{
|
|
|
|
nimble_netif_conn_t *conn = nimble_netif_conn_get(handle);
|
|
|
|
if (conn == NULL) {
|
2021-09-10 21:07:33 +02:00
|
|
|
return -ENOTCONN;
|
2021-02-03 12:11:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int res = ble_hs_hci_read_chan_map(conn->gaphandle, map);
|
|
|
|
if (res != 0) {
|
2021-09-10 21:07:33 +02:00
|
|
|
return -ECANCELED;
|
2021-02-03 12:11:26 +01:00
|
|
|
}
|
|
|
|
|
2021-09-10 21:07:33 +02:00
|
|
|
return 0;
|
2021-02-03 12:11:26 +01:00
|
|
|
}
|
2021-07-01 16:46:35 +02:00
|
|
|
|
|
|
|
int nimble_netif_l2cap_ping(int handle, ble_l2cap_ping_fn cb,
|
|
|
|
const void *data, uint16_t data_len)
|
|
|
|
{
|
|
|
|
nimble_netif_conn_t *conn = nimble_netif_conn_get(handle);
|
|
|
|
if (conn == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return ble_l2cap_ping(conn->gaphandle, cb, data, data_len);
|
|
|
|
}
|