/* * Copyright (C) 2018-2021 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 GNRC netif Implementation * @ingroup pkg_nimble * @brief GNRC netif implementation for NimBLE, enabling the integration * of NimBLE into GNRC * * # About * This NimBLE submodule provides a GNRC netif wrapper for integrating NimBLE * with GNRC and other network stacks using netif (e.g. CCNlite). * * # Concept * According to the IPv6-over-BLE standards (RFC7668 and IPSP), this module * exposes a (configurable) number of point-to-point BLE connections as a single * network device to BLE. Unicast traffic is only send using the corresponding * BLE connection. Multicast and Broadcast packets are duplicated and send via * each open BLE connection. * * # Structure * The netif implementation is able to handle multiple connections * simultaneously. The maximum number of concurrent connections is configured * during compile time, using NimBLEs MYNEWT_VAL_BLE_MAX_CONNECTIONS option. * Dependent on this value, the netif implementation takes care of allocation * all the memory needed. The API of this submodule uses simply integer values * to reference the used connection context (like file descriptors in linux). * * Like any other GNRC network device, the NimBLE netif wrapper runs in its own * thread. This thread is started and configured by the common netif code. All * send and get/set operations are handled by this thread. For efficiency * reasons, receiving of data is however handled completely in the NimBLE host * thread, from where the received data is directly passed on to the * corresponding GNRC thread. * * Although the wrapper hooks into GNRC using the netif interface, it does need * to implement parts of the netdev interface as well. This is done where * needed. * * # Usage * This submodule is designed to work fully asynchronous, in the same way as the * NimBLE interfaces are designed. All functions in this submodule will only * trigger the intended action. Once this action is complete, the module will * report the result asynchronously using the configured callback. * * So before using this module, make sure to register a callback using the * @ref nimble_netif_eventcb() function. * * After this, this module provides functions for managing BLE connections to * other devices. Once these connections are established, this module takes care * of mapping IP packets to the corresponding connections. * * @{ * * @file * @brief GNRC netif implementation for NimBLE * * @author Hauke Petersen */ #ifndef NIMBLE_NETIF_H #define NIMBLE_NETIF_H #include #include #include "net/ble.h" #include "nimble_riot.h" #include "host/ble_hs.h" #ifdef __cplusplus extern "C" { #endif /** * @brief The maximum number of BLE connections that can be open concurrently * * @note This value *must* be <= MYNEWT_VAL_BLE_MAX_CONNECTIONS */ #ifndef NIMBLE_NETIF_MAX_CONN #define NIMBLE_NETIF_MAX_CONN (MYNEWT_VAL_BLE_MAX_CONNECTIONS) #endif /** * @brief Default L2CAP channel ID to use */ #ifndef NIMBLE_NETIF_CID #define NIMBLE_NETIF_CID (BLE_L2CAP_CID_IPSP) #endif /** * @brief Default MTU size supported by the NimBLE netif wrapper */ /* NOTE: We do not use the @ref IPV6_MIN_MTU define here, as the iov6.h header pulls in some other RIOT headers that clash with NimBLE header (e.g. * byteorder.h vs. endian.h) */ #ifndef NIMBLE_NETIF_MTU #define NIMBLE_NETIF_MTU (1280U) #endif /** * @brief Flags for enabling legacy advertisement and high-duty cycle mode * when accepting incoming connections */ enum { NIMBLE_NETIF_FLAG_LEGACY = 0x01, /**< use legacy advertising mode */ NIMBLE_NETIF_FLAG_HD_MODE = 0x02, /**< use high duty cycle mode, only * valid for direct advertising */ }; /** * @brief Parameter set used to configure accepting connections (advertising) */ typedef struct { uint8_t flags; /**< flags */ uint8_t channel_map; /**< specify custom channel map */ uint8_t own_addr_type; /**< specify our own address type to use */ int8_t tx_power; /**< specify TX power to be used */ uint32_t adv_itvl_ms; /**< advertising interval [ms] */ uint32_t timeout_ms; /**< stop accepting after this time [ms] */ nimble_phy_t primary_phy; /**< primary PHY mode */ nimble_phy_t secondary_phy; /**< secondary PHY mode */ } nimble_netif_accept_cfg_t; /** * @brief Parameter set used to configure connection initiation */ typedef struct { uint16_t scan_itvl_ms; /**< scan interval [ms] */ uint16_t scan_window_ms; /**< scan window [ms] */ uint16_t conn_itvl_min_ms; /**< connection interval, lower bound [ms] */ uint16_t conn_itvl_max_ms; /**< connection interval, upper bound [ms] */ uint16_t conn_supervision_timeout_ms; /**< supervision timeout [ms] */ uint16_t conn_slave_latency;/**< slave latency */ uint32_t timeout_ms; /**< abort connection initiation after this time * [ms] */ uint8_t phy_mode; /**< PHY mode used for the connection */ uint8_t own_addr_type; /**< specify our own address type to use */ } nimble_netif_connect_cfg_t; /** * @brief Set to > 0 to enforce different connection intervals for each of the * nodes BLE connections * * Enabling this option will enforce that every BLE connection a node maintains, * independent of the nodes role, uses a different connection interval. The * value of NIMBLE_NETIF_CONN_ITVL_SPACING specifies the minimum spacing between * connection intervals as multiple of 1,25ms. E.g. a value of 2 will force each * connection to use a connection interval that is at least 2.5ms different from * all other used connection intervals. * * If a node is the coordinator of a connection, it will generate a connection * interval for each new connection based on a random value by adhering to the * spacing constraint. * * If a node is the subordinate of a new connection, it will check if the given * connection interval is fulfilling the spacing constraint with respect to * already existing connections of that node. If the connection interval of the * new connection is not properly spaced, the node will drop the connection * right away, giving the coordinator node the possibly to reconnect with a * different connection interval. */ #ifndef NIMBLE_NETIF_CONN_ITVL_SPACING #define NIMBLE_NETIF_CONN_ITVL_SPACING 0 #endif /** * @brief Event types triggered by the NimBLE netif module */ typedef enum { NIMBLE_NETIF_ACCEPTING, /**< accepting incoming connections */ NIMBLE_NETIF_ACCEPT_STOP, /**< stop accepting incoming connections */ NIMBLE_NETIF_INIT_MASTER, /**< conn. procedure started (as master) */ NIMBLE_NETIF_INIT_SLAVE, /**< conn. procedure started (as slave) */ NIMBLE_NETIF_CONNECTED_MASTER, /**< connection established as master */ NIMBLE_NETIF_CONNECTED_SLAVE, /**< connection established as slave */ NIMBLE_NETIF_CLOSED_MASTER, /**< connection closed (we were master) */ NIMBLE_NETIF_CLOSED_SLAVE, /**< connection closed (we were slave) */ NIMBLE_NETIF_ABORT_MASTER, /**< connection est. abort (as master) */ NIMBLE_NETIF_ABORT_SLAVE, /**< connection est. abort (as slave) */ NIMBLE_NETIF_CONN_UPDATED, /**< connection parameter update done */ } nimble_netif_event_t; /** * @brief Flags describing the state of a single connection context */ enum { NIMBLE_NETIF_L2CAP_CLIENT = 0x0001, /**< L2CAP client */ NIMBLE_NETIF_L2CAP_SERVER = 0x0002, /**< L2CAP server */ NIMBLE_NETIF_L2CAP_CONNECTED = 0x0003, /**< L2CAP is connected */ NIMBLE_NETIF_GAP_MASTER = 0x0010, /**< GAP master */ NIMBLE_NETIF_GAP_SLAVE = 0x0020, /**< GAP slave */ NIMBLE_NETIF_GAP_CONNECTED = 0x0030, /**< GAP is connected */ NIMBLE_NETIF_ADV = 0x0100, /**< currently advertising */ NIMBLE_NETIF_CONNECTING = 0x4000, /**< connection in progress */ NIMBLE_NETIF_UNUSED = 0x8000, /**< context unused */ NIMBLE_NETIF_ANY = 0xffff, /**< match any state */ }; /** * @brief Event callback signature used for asynchronous event signaling * * @note The event callback is always executed in NimBLE's host thread * * @param[in] handle handle to the connection that triggered 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, const uint8_t *addr); /** * @brief Initialize the netif implementation, spawns the netif thread * * This function is meant to be called once during system initialization, i.e. * auto-init. */ void nimble_netif_init(void); /** * @brief Register a global event callback, servicing all NimBLE connections * * @note The event callback is always executed in NimBLE's host thread * * @param[in] cb event callback to register, may be NULL */ void nimble_netif_eventcb(nimble_netif_eventcb_t cb); /** * @brief Open a BLE connection as BLE master * * @param[in] addr address of the advertising BLE slave, in the NimBLE * addr format (little endian) * @param[in] cfg connection parameters * * @return the used connection handle on success * @return -EBUSY if already connected to the given address or if a connection * setup procedure is in progress * @return -ENOMEM if no connection context memory is available * @return -ECANCELED if unable to find valid connection interval * @return -EINVAL if unable to apply given PHY mode * @return -EIO on all other NimBLE errors */ int nimble_netif_connect(const ble_addr_t *addr, const nimble_netif_connect_cfg_t *cfg); /** * @brief Close the connection with the given handle * * @param[in] handle handle for the connection to be closed * * @return 0 on success * @return -EINVAL if the handle is invalid * @return -ENOTCONN if context for given handle is not connected * @return -EIO on all other NimBLE errors */ int nimble_netif_close(int handle); /** * @brief Accept incoming connections by starting to advertise this node * * @param[in] ad advertising data (in BLE AD format) * @param[in] ad_len length of @p ad in bytes * @param[in] cfg advertising parameters to use * * @return 0 on success * @return -EALREADY if already advertising * @return -ENOMEM on insufficient connection memory * @return -EINVAL on invalid configuration parameters * @return -ECANCELED on other errors */ int nimble_netif_accept(const uint8_t *ad, size_t ad_len, const nimble_netif_accept_cfg_t *cfg); /** * @brief Wait for an incoming connection from a specific peer, sending * directed advertisements * * @param[in] addr BLE address of the target peer * @param[in] cfg advertising parameters to use * * @return 0 on success * @return -EALREADY if already advertising * @return -ENOMEM on insufficient connection memory * @return -EINVAL on invalid configuration parameters * @return -ECANCELED on other errors */ int nimble_netif_accept_direct(const ble_addr_t *addr, const nimble_netif_accept_cfg_t *cfg); /** * @brief Stop accepting incoming connections (stop advertising) * * * @return 0 on success * @return -EALREADY if not currently advertising * @return -EIO on other NimBLE errors */ int nimble_netif_accept_stop(void); /** * @brief Update the connection parameters for the given connection * * @param[in] handle connection handle * @param[in] conn_params new connection parameters to apply * * @return 0 on success * @return -ENOTCONN if handle does not point to a connection * @return -ECANCELED if applying the given parameters failed */ int nimble_netif_update(int handle, const struct ble_gap_upd_params *conn_params); /** * @brief Get the currently used channel map for the given connection as * bitmap * * @param[in] handle connection handle * @param[out] map used channel map, map[0] denotes channels 0 to 7, * map[1] chan 8-15, ..., map[5] channels 33 to 36. * **must** be able to hold 5 bytes. * * @return 0 on success * @return -ENOTCONN if handle dos not point to a connection * @return -ECANCELED if reading the channel map failed otherwise */ int nimble_netif_used_chanmap(int handle, uint8_t map[5]); #ifdef __cplusplus } #endif #endif /* NIMBLE_NETIF_H */ /** @} */