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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @defgroup pkg_nimble_netif_conn Connection State Management for netif
|
|
|
|
* @ingroup pkg_nimble_netif
|
|
|
|
* @brief Helper module for managing the memory needed to store the
|
|
|
|
* BLE connection state for the netif wrapper
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Connection allocation and maintenance for NimBLE netif
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef NIMBLE_NETIF_CONN_H
|
|
|
|
#define NIMBLE_NETIF_CONN_H
|
|
|
|
|
|
|
|
#include <stdint.h>
|
2021-04-21 20:33:41 +02:00
|
|
|
#include <stdbool.h>
|
2019-05-17 11:15:49 +02:00
|
|
|
|
|
|
|
#include "nimble_netif.h"
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Value for marking a handle invalid
|
|
|
|
*/
|
|
|
|
#define NIMBLE_NETIF_CONN_INVALID (-1)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Memory layout for holding the relevant connection information
|
|
|
|
*/
|
|
|
|
typedef struct {
|
|
|
|
struct ble_l2cap_chan *coc; /**< l2cap context as exposed by NimBLE */
|
|
|
|
uint16_t gaphandle; /**< GAP handle exposed by NimBLE */
|
|
|
|
uint16_t state; /**< the current state of the context */
|
|
|
|
uint8_t addr[BLE_ADDR_LEN]; /**< BLE address of connected peer
|
|
|
|
(in network byte order) */
|
|
|
|
} nimble_netif_conn_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Iterator function signature used by @ref nimble_netif_conn_foreach()
|
|
|
|
*
|
|
|
|
* @param[in] conn connection context of the current entry
|
|
|
|
* @param[in] handle handle of the current entry
|
|
|
|
* @param[in] arg user supplied argument
|
|
|
|
*
|
|
|
|
* @return 0 to continue
|
|
|
|
* @return != 0 to stop iterating
|
|
|
|
*/
|
|
|
|
typedef int (*nimble_netif_conn_iter_t)(nimble_netif_conn_t *conn,
|
|
|
|
int handle, void *arg);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize the connection state manager
|
|
|
|
*
|
|
|
|
* This functions is typically called by @ref nimble_netif_init().
|
|
|
|
*/
|
|
|
|
void nimble_netif_conn_init(void);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Get the connection context corresponding to the given handle
|
|
|
|
*
|
|
|
|
* @param[in] handle handle to a connection context
|
|
|
|
*
|
|
|
|
* @return pointer to the corresponding connection context
|
|
|
|
* @return NULL if handle in invalid
|
|
|
|
*/
|
|
|
|
nimble_netif_conn_t *nimble_netif_conn_get(int handle);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Get the handle to the context that is currently advertising
|
|
|
|
*
|
|
|
|
* @return handle to the currently advertising context
|
|
|
|
* @return NIMBLE_NETIF_CONN_INVALID if not advertising
|
|
|
|
*/
|
|
|
|
int nimble_netif_conn_get_adv(void);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Get the handle to the context that is busy connecting
|
|
|
|
*
|
|
|
|
* @return handle to the busy context
|
|
|
|
* @return NIMBLE_NETIF_CONN_INVALID if not busy connecting
|
|
|
|
*/
|
|
|
|
int nimble_netif_conn_get_connecting(void);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Find the connection to the peer with the given BLE address
|
|
|
|
*
|
|
|
|
* @param[in] addr BLE address, in network byte order
|
|
|
|
*
|
|
|
|
* @return handle to the matching connection context
|
|
|
|
* @return NIMBLE_NETIF_CONN_INVALID if no matching connection was found
|
|
|
|
*/
|
|
|
|
int nimble_netif_conn_get_by_addr(const uint8_t *addr);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Find the connection using the given NimBLE GAP handle
|
|
|
|
*
|
|
|
|
* @param[in] gaphandle GAP handle as exposed by NimBLE
|
|
|
|
*
|
|
|
|
* @return handle to the matching connection context
|
|
|
|
* @return NIMBLE_NETIF_CONN_INVALID if no matching connection was found
|
|
|
|
*/
|
|
|
|
int nimble_netif_conn_get_by_gaphandle(uint16_t gaphandle);
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Iterate over all connection contexts that match the filter condition
|
|
|
|
*
|
|
|
|
* @warning Do not call any other nimble_netif_conn function from within the
|
|
|
|
* callback, this will lead to a deadlock!
|
|
|
|
*
|
|
|
|
* @param[in] filter filter mask
|
|
|
|
* @param[in] cb callback called on each filtered entry
|
|
|
|
* @param[in] arg user argument
|
|
|
|
*/
|
|
|
|
void nimble_netif_conn_foreach(uint16_t filter,
|
|
|
|
nimble_netif_conn_iter_t cb, void *arg);
|
|
|
|
|
2020-04-27 15:15:50 +02:00
|
|
|
/**
|
|
|
|
* @brief Find the next context that matches the filter condition
|
|
|
|
*
|
|
|
|
* This function allows for iterating connection contexts in a non-blocking way.
|
|
|
|
*
|
|
|
|
* @param[in] handle last used handle, set to NIMBLE_NETIF_CONN_INVALID
|
|
|
|
to get the first matching entry
|
|
|
|
* @param[in] filter filter mask
|
|
|
|
*
|
|
|
|
* @return handle of the next matching connection context
|
|
|
|
* @return NIMBLE_NETIF_CONN_INVALID if no matching context was found
|
|
|
|
*/
|
|
|
|
int nimble_netif_conn_get_next(int handle, uint16_t filter);
|
|
|
|
|
2019-05-17 11:15:49 +02:00
|
|
|
/**
|
|
|
|
* @brief Count the number of connections contexts for which the given filter
|
|
|
|
* applies
|
|
|
|
*
|
|
|
|
* @param[in] filter filter mask
|
|
|
|
*
|
|
|
|
* @return number of contexts for which the filter applied
|
|
|
|
*/
|
|
|
|
|
|
|
|
unsigned nimble_netif_conn_count(uint16_t filter);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Allocate an unused context for starting a connection
|
|
|
|
*
|
|
|
|
* @param[in] addr the BLE address of the peer node, in network byte
|
|
|
|
* order
|
|
|
|
*
|
|
|
|
* @return handle used for the new connection
|
|
|
|
*/
|
|
|
|
int nimble_netif_conn_start_connection(const uint8_t *addr);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Reserve a unused context for the purpose of accepting a new
|
|
|
|
* connection
|
|
|
|
*
|
|
|
|
* @return handle of the reserved context
|
|
|
|
* @return NIMBLE_NETIF_CONN_INVALID if no unused context was available
|
|
|
|
*/
|
|
|
|
int nimble_netif_conn_start_adv(void);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Free the connection context with the given handle
|
|
|
|
*/
|
2019-12-18 16:17:16 +01:00
|
|
|
void nimble_netif_conn_free(int handle, uint8_t *addr);
|
2019-05-17 11:15:49 +02:00
|
|
|
|
2021-04-21 20:33:41 +02:00
|
|
|
/**
|
|
|
|
* @brief Get the used connection interval for the given connection handle
|
|
|
|
*
|
|
|
|
* @param[in] handle connection handle
|
|
|
|
*
|
|
|
|
* @return used connection interval on success, multiples of 1.25ms
|
|
|
|
* @return 0 if unable to get connection interval
|
|
|
|
*/
|
|
|
|
uint16_t nimble_netif_conn_get_itvl(int handle);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Check if the given connection interval is used, taking the minimal
|
|
|
|
* spacing as defined by NIMBLE_NETIF_CONN_ITVL_SPACING into account
|
|
|
|
*
|
|
|
|
* @param[in] itvl connection interval to check, multiples of 1.25ms
|
|
|
|
* @param[in] skip_handle do not compare against connection interval for this
|
|
|
|
* handle, set to NIMBLE_NETIF_CONN_INVALID to check
|
|
|
|
* all
|
|
|
|
*
|
|
|
|
* @return true if given interval is used
|
|
|
|
* @return false if given interval is not used
|
|
|
|
*/
|
|
|
|
bool nimble_netif_conn_itvl_used(uint16_t itvl, int skip_handle);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Check if connection interval used by the given connection is valid
|
|
|
|
*
|
|
|
|
* @param[in] handle connection to verify
|
|
|
|
*
|
|
|
|
* @return true if the connection interval of the given connection collides
|
|
|
|
* with the connection interval of another BLE connection
|
|
|
|
* @return false if the connection interval of the given connection is valid
|
|
|
|
*/
|
|
|
|
bool nimble_netif_conn_itvl_invalid(int handle);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Generate a pseudorandom connection interval from the given range
|
|
|
|
*
|
|
|
|
* If the NIMBLE_NETIF_CONN_ITVL_SPACING option is enabled, this function
|
|
|
|
* ensures that the generated connection interval is spaced at least
|
|
|
|
* NIMBLE_NETIF_CONN_ITVL_SPACING from the connection interval of each open
|
|
|
|
* BLE connection.
|
|
|
|
*
|
|
|
|
* @param[in] min minimum connection interval
|
|
|
|
* @param[in] max maximum connection interval
|
|
|
|
*
|
|
|
|
* @return generated connection interval on success, multiples of 1.25ms
|
|
|
|
* @return 0 if no valid connection interval could be generated
|
|
|
|
*/
|
|
|
|
uint16_t nimble_netif_conn_gen_itvl(uint16_t min, uint16_t max);
|
|
|
|
|
2019-05-17 11:15:49 +02:00
|
|
|
/**
|
|
|
|
* @brief Find the connection context with a given GAP handle and return a
|
|
|
|
* pointer to it
|
|
|
|
*
|
|
|
|
* @param[in] gh GAP handle used by NimBLE
|
|
|
|
*
|
|
|
|
* @return Pointer to the selected context
|
|
|
|
* @return NULL if no fitting context was found
|
|
|
|
*/
|
|
|
|
static inline
|
|
|
|
nimble_netif_conn_t *nimble_netif_conn_from_gaphandle(uint16_t gh)
|
|
|
|
{
|
|
|
|
return nimble_netif_conn_get(nimble_netif_conn_get_by_gaphandle(gh));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Convenience function to check if any context is currently in the
|
|
|
|
* connecting state (@ref NIMBLE_NETIF_CONNECTING)
|
|
|
|
*
|
|
|
|
* @return != 0 if true
|
|
|
|
* @return 0 if false
|
|
|
|
*/
|
|
|
|
static inline int nimble_netif_conn_connecting(void)
|
|
|
|
{
|
|
|
|
return (nimble_netif_conn_get_connecting() != NIMBLE_NETIF_CONN_INVALID);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Convenience function to check if we are currently connected to a
|
|
|
|
* peer with the given address
|
|
|
|
*
|
|
|
|
* @param[in] addr BLE address, in network byte order
|
|
|
|
*
|
|
|
|
* @return != 0 if true
|
|
|
|
* @return 0 if false
|
|
|
|
*/
|
|
|
|
static inline int nimble_netif_conn_connected(const uint8_t *addr)
|
|
|
|
{
|
|
|
|
return (nimble_netif_conn_get_by_addr(addr) != NIMBLE_NETIF_CONN_INVALID);
|
|
|
|
}
|
|
|
|
|
2020-09-03 11:42:08 +02:00
|
|
|
/**
|
|
|
|
* @brief Test if the given connection is (still) open
|
|
|
|
*
|
|
|
|
* @param[in] conn Connection to test
|
|
|
|
*
|
|
|
|
* @return != 0 if true
|
|
|
|
* @return 0 if false
|
|
|
|
*/
|
|
|
|
static inline int nimble_netif_conn_is_open(const nimble_netif_conn_t *conn)
|
|
|
|
{
|
|
|
|
return (conn->coc != NULL);
|
|
|
|
}
|
2019-05-17 11:15:49 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Convenience function to check if any context is currently in the
|
|
|
|
* advertising state (@ref NIMBLE_NETIF_ADV)
|
|
|
|
*
|
|
|
|
* @return != 0 if true
|
|
|
|
* @return 0 if false
|
|
|
|
*/
|
|
|
|
static inline int nimble_netif_conn_is_adv(void)
|
|
|
|
{
|
|
|
|
return (nimble_netif_conn_get_adv() != NIMBLE_NETIF_CONN_INVALID);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* NIMBLE_NETIF_CONN_H */
|
|
|
|
/** @} */
|