mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
470 lines
12 KiB
C
470 lines
12 KiB
C
/*
|
|
* Copyright 2019 Espressif Systems (Shanghai) PTE LTD
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one
|
|
* or more contributor license agreements. See the NOTICE file
|
|
* distributed with this work for additional information
|
|
* regarding copyright ownership. The ASF licenses this file
|
|
* to you under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance
|
|
* with the License. You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing,
|
|
* software distributed under the License is distributed on an
|
|
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
* KIND, either express or implied. See the License for the
|
|
* specific language governing permissions and limitations
|
|
* under the License.
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include "sysinit/sysinit.h"
|
|
#include "nimble/hci_common.h"
|
|
#include "host/ble_hs.h"
|
|
#include "nimble/nimble_port.h"
|
|
#include "nimble/nimble_port_freertos.h"
|
|
#include "esp_nimble_hci.h"
|
|
#include "esp_bt.h"
|
|
#include "freertos/semphr.h"
|
|
|
|
#define NIMBLE_VHCI_TIMEOUT_MS 2000
|
|
|
|
static ble_hci_trans_rx_cmd_fn *ble_hci_rx_cmd_hs_cb;
|
|
static void *ble_hci_rx_cmd_hs_arg;
|
|
|
|
static ble_hci_trans_rx_acl_fn *ble_hci_rx_acl_hs_cb;
|
|
static void *ble_hci_rx_acl_hs_arg;
|
|
|
|
static struct os_mbuf_pool ble_hci_acl_mbuf_pool;
|
|
static struct os_mempool_ext ble_hci_acl_pool;
|
|
/*
|
|
* The MBUF payload size must accommodate the HCI data header size plus the
|
|
* maximum ACL data packet length. The ACL block size is the size of the
|
|
* mbufs we will allocate.
|
|
*/
|
|
#define ACL_BLOCK_SIZE OS_ALIGN(MYNEWT_VAL(BLE_ACL_BUF_SIZE) \
|
|
+ BLE_MBUF_MEMBLOCK_OVERHEAD \
|
|
+ BLE_HCI_DATA_HDR_SZ, OS_ALIGNMENT)
|
|
|
|
static os_membuf_t ble_hci_acl_buf[
|
|
OS_MEMPOOL_SIZE(MYNEWT_VAL(BLE_ACL_BUF_COUNT),
|
|
ACL_BLOCK_SIZE)];
|
|
|
|
static struct os_mempool ble_hci_cmd_pool;
|
|
static os_membuf_t ble_hci_cmd_buf[
|
|
OS_MEMPOOL_SIZE(1, BLE_HCI_TRANS_CMD_SZ)
|
|
];
|
|
|
|
static struct os_mempool ble_hci_evt_hi_pool;
|
|
static os_membuf_t ble_hci_evt_hi_buf[
|
|
OS_MEMPOOL_SIZE(MYNEWT_VAL(BLE_HCI_EVT_HI_BUF_COUNT),
|
|
MYNEWT_VAL(BLE_HCI_EVT_BUF_SIZE))
|
|
];
|
|
|
|
static struct os_mempool ble_hci_evt_lo_pool;
|
|
static os_membuf_t ble_hci_evt_lo_buf[
|
|
OS_MEMPOOL_SIZE(MYNEWT_VAL(BLE_HCI_EVT_LO_BUF_COUNT),
|
|
MYNEWT_VAL(BLE_HCI_EVT_BUF_SIZE))
|
|
];
|
|
|
|
static SemaphoreHandle_t vhci_send_sem;
|
|
const static char *TAG = "NimBLE";
|
|
|
|
void ble_hci_trans_cfg_hs(ble_hci_trans_rx_cmd_fn *cmd_cb,
|
|
void *cmd_arg,
|
|
ble_hci_trans_rx_acl_fn *acl_cb,
|
|
void *acl_arg)
|
|
{
|
|
ble_hci_rx_cmd_hs_cb = cmd_cb;
|
|
ble_hci_rx_cmd_hs_arg = cmd_arg;
|
|
ble_hci_rx_acl_hs_cb = acl_cb;
|
|
ble_hci_rx_acl_hs_arg = acl_arg;
|
|
}
|
|
|
|
|
|
int ble_hci_trans_hs_cmd_tx(uint8_t *cmd)
|
|
{
|
|
uint16_t len;
|
|
uint8_t rc = 0;
|
|
|
|
assert(cmd != NULL);
|
|
*cmd = BLE_HCI_UART_H4_CMD;
|
|
len = BLE_HCI_CMD_HDR_LEN + cmd[3] + 1;
|
|
if (!esp_vhci_host_check_send_available()) {
|
|
ESP_LOGD(TAG, "Controller not ready to receive packets");
|
|
}
|
|
|
|
if (xSemaphoreTake(vhci_send_sem, NIMBLE_VHCI_TIMEOUT_MS / portTICK_PERIOD_MS) == pdTRUE) {
|
|
esp_vhci_host_send_packet(cmd, len);
|
|
} else {
|
|
rc = BLE_HS_ETIMEOUT_HCI;
|
|
}
|
|
|
|
ble_hci_trans_buf_free(cmd);
|
|
return rc;
|
|
}
|
|
|
|
int ble_hci_trans_ll_evt_tx(uint8_t *hci_ev)
|
|
{
|
|
int rc = ESP_FAIL;
|
|
|
|
if (ble_hci_rx_cmd_hs_cb) {
|
|
rc = ble_hci_rx_cmd_hs_cb(hci_ev, ble_hci_rx_cmd_hs_arg);
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
int ble_hci_trans_hs_acl_tx(struct os_mbuf *om)
|
|
{
|
|
uint16_t len = 0;
|
|
uint8_t data[MYNEWT_VAL(BLE_ACL_BUF_SIZE) + 1], rc = 0;
|
|
/* If this packet is zero length, just free it */
|
|
if (OS_MBUF_PKTLEN(om) == 0) {
|
|
os_mbuf_free_chain(om);
|
|
return 0;
|
|
}
|
|
data[0] = BLE_HCI_UART_H4_ACL;
|
|
len++;
|
|
|
|
if (!esp_vhci_host_check_send_available()) {
|
|
ESP_LOGD(TAG, "Controller not ready to receive packets");
|
|
}
|
|
|
|
os_mbuf_copydata(om, 0, OS_MBUF_PKTLEN(om), &data[1]);
|
|
len += OS_MBUF_PKTLEN(om);
|
|
|
|
if (xSemaphoreTake(vhci_send_sem, NIMBLE_VHCI_TIMEOUT_MS / portTICK_PERIOD_MS) == pdTRUE) {
|
|
esp_vhci_host_send_packet(data, len);
|
|
} else {
|
|
rc = BLE_HS_ETIMEOUT_HCI;
|
|
}
|
|
|
|
os_mbuf_free_chain(om);
|
|
|
|
return rc;
|
|
}
|
|
|
|
int ble_hci_trans_ll_acl_tx(struct os_mbuf *om)
|
|
{
|
|
int rc = ESP_FAIL;
|
|
|
|
if (ble_hci_rx_acl_hs_cb) {
|
|
rc = ble_hci_rx_acl_hs_cb(om, ble_hci_rx_acl_hs_arg);
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
uint8_t *ble_hci_trans_buf_alloc(int type)
|
|
{
|
|
uint8_t *buf;
|
|
|
|
switch (type) {
|
|
case BLE_HCI_TRANS_BUF_CMD:
|
|
buf = os_memblock_get(&ble_hci_cmd_pool);
|
|
break;
|
|
|
|
case BLE_HCI_TRANS_BUF_EVT_HI:
|
|
buf = os_memblock_get(&ble_hci_evt_hi_pool);
|
|
if (buf == NULL) {
|
|
/* If no high-priority event buffers remain, try to grab a
|
|
* low-priority one.
|
|
*/
|
|
buf = ble_hci_trans_buf_alloc(BLE_HCI_TRANS_BUF_EVT_LO);
|
|
}
|
|
break;
|
|
|
|
case BLE_HCI_TRANS_BUF_EVT_LO:
|
|
buf = os_memblock_get(&ble_hci_evt_lo_pool);
|
|
break;
|
|
|
|
default:
|
|
assert(0);
|
|
buf = NULL;
|
|
}
|
|
|
|
return buf;
|
|
}
|
|
|
|
void ble_hci_trans_buf_free(uint8_t *buf)
|
|
{
|
|
int rc;
|
|
/* XXX: this may look a bit odd, but the controller uses the command
|
|
* buffer to send back the command complete/status as an immediate
|
|
* response to the command. This was done to insure that the controller
|
|
* could always send back one of these events when a command was received.
|
|
* Thus, we check to see which pool the buffer came from so we can free
|
|
* it to the appropriate pool
|
|
*/
|
|
if (os_memblock_from(&ble_hci_evt_hi_pool, buf)) {
|
|
rc = os_memblock_put(&ble_hci_evt_hi_pool, buf);
|
|
assert(rc == 0);
|
|
} else if (os_memblock_from(&ble_hci_evt_lo_pool, buf)) {
|
|
rc = os_memblock_put(&ble_hci_evt_lo_pool, buf);
|
|
assert(rc == 0);
|
|
} else {
|
|
assert(os_memblock_from(&ble_hci_cmd_pool, buf));
|
|
rc = os_memblock_put(&ble_hci_cmd_pool, buf);
|
|
assert(rc == 0);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Unsupported; the RAM transport does not have a dedicated ACL data packet
|
|
* pool.
|
|
*/
|
|
int ble_hci_trans_set_acl_free_cb(os_mempool_put_fn *cb, void *arg)
|
|
{
|
|
return BLE_ERR_UNSUPPORTED;
|
|
}
|
|
|
|
int ble_hci_trans_reset(void)
|
|
{
|
|
/* No work to do. All allocated buffers are owned by the host or
|
|
* controller, and they will get freed by their owners.
|
|
*/
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Allocates a buffer (mbuf) for ACL operation.
|
|
*
|
|
* @return The allocated buffer on success;
|
|
* NULL on buffer exhaustion.
|
|
*/
|
|
static struct os_mbuf *ble_hci_trans_acl_buf_alloc(void)
|
|
{
|
|
struct os_mbuf *m;
|
|
uint8_t usrhdr_len;
|
|
|
|
#if MYNEWT_VAL(BLE_DEVICE)
|
|
usrhdr_len = sizeof(struct ble_mbuf_hdr);
|
|
#elif MYNEWT_VAL(BLE_HS_FLOW_CTRL)
|
|
usrhdr_len = BLE_MBUF_HS_HDR_LEN;
|
|
#else
|
|
usrhdr_len = 0;
|
|
#endif
|
|
|
|
m = os_mbuf_get_pkthdr(&ble_hci_acl_mbuf_pool, usrhdr_len);
|
|
return m;
|
|
}
|
|
|
|
static void ble_hci_rx_acl(uint8_t *data, uint16_t len)
|
|
{
|
|
struct os_mbuf *m;
|
|
int sr;
|
|
if (len < BLE_HCI_DATA_HDR_SZ || len > MYNEWT_VAL(BLE_ACL_BUF_SIZE)) {
|
|
return;
|
|
}
|
|
|
|
m = ble_hci_trans_acl_buf_alloc();
|
|
|
|
if (!m) {
|
|
return;
|
|
}
|
|
if (os_mbuf_append(m, data, len)) {
|
|
os_mbuf_free_chain(m);
|
|
return;
|
|
}
|
|
OS_ENTER_CRITICAL(sr);
|
|
if (ble_hci_rx_acl_hs_cb) {
|
|
ble_hci_rx_acl_hs_cb(m, NULL);
|
|
}
|
|
OS_EXIT_CRITICAL(sr);
|
|
}
|
|
|
|
static void ble_hci_transport_init(void)
|
|
{
|
|
int rc;
|
|
|
|
/* Ensure this function only gets called by sysinit. */
|
|
SYSINIT_ASSERT_ACTIVE();
|
|
|
|
rc = os_mempool_ext_init(&ble_hci_acl_pool,
|
|
MYNEWT_VAL(BLE_ACL_BUF_COUNT),
|
|
ACL_BLOCK_SIZE,
|
|
ble_hci_acl_buf,
|
|
"ble_hci_acl_pool");
|
|
SYSINIT_PANIC_ASSERT(rc == 0);
|
|
|
|
rc = os_mbuf_pool_init(&ble_hci_acl_mbuf_pool,
|
|
&ble_hci_acl_pool.mpe_mp,
|
|
ACL_BLOCK_SIZE,
|
|
MYNEWT_VAL(BLE_ACL_BUF_COUNT));
|
|
SYSINIT_PANIC_ASSERT(rc == 0);
|
|
|
|
/*
|
|
* Create memory pool of HCI command buffers. NOTE: we currently dont
|
|
* allow this to be configured. The controller will only allow one
|
|
* outstanding command. We decided to keep this a pool in case we allow
|
|
* allow the controller to handle more than one outstanding command.
|
|
*/
|
|
rc = os_mempool_init(&ble_hci_cmd_pool,
|
|
1,
|
|
BLE_HCI_TRANS_CMD_SZ,
|
|
ble_hci_cmd_buf,
|
|
"ble_hci_cmd_pool");
|
|
SYSINIT_PANIC_ASSERT(rc == 0);
|
|
|
|
rc = os_mempool_init(&ble_hci_evt_hi_pool,
|
|
MYNEWT_VAL(BLE_HCI_EVT_HI_BUF_COUNT),
|
|
MYNEWT_VAL(BLE_HCI_EVT_BUF_SIZE),
|
|
ble_hci_evt_hi_buf,
|
|
"ble_hci_evt_hi_pool");
|
|
SYSINIT_PANIC_ASSERT(rc == 0);
|
|
|
|
rc = os_mempool_init(&ble_hci_evt_lo_pool,
|
|
MYNEWT_VAL(BLE_HCI_EVT_LO_BUF_COUNT),
|
|
MYNEWT_VAL(BLE_HCI_EVT_BUF_SIZE),
|
|
ble_hci_evt_lo_buf,
|
|
"ble_hci_evt_lo_pool");
|
|
SYSINIT_PANIC_ASSERT(rc == 0);
|
|
}
|
|
|
|
/*
|
|
* @brief: BT controller callback function, used to notify the upper layer that
|
|
* controller is ready to receive command
|
|
*/
|
|
static void controller_rcv_pkt_ready(void)
|
|
{
|
|
if (vhci_send_sem) {
|
|
xSemaphoreGive(vhci_send_sem);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* @brief: BT controller callback function, to transfer data packet to the host
|
|
*/
|
|
static int host_rcv_pkt(uint8_t *data, uint16_t len)
|
|
{
|
|
|
|
if (data[0] == BLE_HCI_UART_H4_EVT) {
|
|
uint8_t *evbuf;
|
|
int totlen;
|
|
int rc;
|
|
|
|
totlen = BLE_HCI_EVENT_HDR_LEN + data[2];
|
|
assert(totlen <= UINT8_MAX + BLE_HCI_EVENT_HDR_LEN);
|
|
|
|
if (data[1] == BLE_HCI_EVCODE_HW_ERROR) {
|
|
assert(0);
|
|
}
|
|
|
|
/* Allocate LE Advertising Report Event from lo pool only */
|
|
if ((data[1] == BLE_HCI_EVCODE_LE_META) && (data[3] == BLE_HCI_LE_SUBEV_ADV_RPT)) {
|
|
evbuf = ble_hci_trans_buf_alloc(BLE_HCI_TRANS_BUF_EVT_LO);
|
|
/* Skip advertising report if we're out of memory */
|
|
if (!evbuf) {
|
|
return 0;
|
|
}
|
|
} else {
|
|
evbuf = ble_hci_trans_buf_alloc(BLE_HCI_TRANS_BUF_EVT_HI);
|
|
assert(evbuf != NULL);
|
|
}
|
|
|
|
memcpy(evbuf, &data[1], totlen);
|
|
|
|
rc = ble_hci_trans_ll_evt_tx(evbuf);
|
|
assert(rc == 0);
|
|
} else if (data[0] == BLE_HCI_UART_H4_ACL) {
|
|
ble_hci_rx_acl(data + 1, len - 1);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static const esp_vhci_host_callback_t vhci_host_cb = {
|
|
.notify_host_send_available = controller_rcv_pkt_ready,
|
|
.notify_host_recv = host_rcv_pkt,
|
|
};
|
|
|
|
|
|
esp_err_t esp_nimble_hci_init(void)
|
|
{
|
|
esp_err_t ret;
|
|
if ((ret = esp_vhci_host_register_callback(&vhci_host_cb)) != ESP_OK) {
|
|
return ret;
|
|
}
|
|
|
|
ble_hci_transport_init();
|
|
|
|
vhci_send_sem = xSemaphoreCreateBinary();
|
|
if (vhci_send_sem == NULL) {
|
|
return ESP_ERR_NO_MEM;
|
|
}
|
|
|
|
xSemaphoreGive(vhci_send_sem);
|
|
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t esp_nimble_hci_and_controller_init(void)
|
|
{
|
|
esp_err_t ret;
|
|
|
|
esp_bt_controller_mem_release(ESP_BT_MODE_CLASSIC_BT);
|
|
|
|
esp_bt_controller_config_t bt_cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT();
|
|
|
|
if ((ret = esp_bt_controller_init(&bt_cfg)) != ESP_OK) {
|
|
return ret;
|
|
}
|
|
|
|
if ((ret = esp_bt_controller_enable(ESP_BT_MODE_BLE)) != ESP_OK) {
|
|
return ret;
|
|
}
|
|
return esp_nimble_hci_init();
|
|
}
|
|
|
|
static esp_err_t ble_hci_transport_deinit(void)
|
|
{
|
|
int ret = 0;
|
|
|
|
ret += os_mempool_clear(&ble_hci_evt_lo_pool);
|
|
|
|
ret += os_mempool_clear(&ble_hci_evt_hi_pool);
|
|
|
|
ret += os_mempool_clear(&ble_hci_cmd_pool);
|
|
|
|
ret += os_mempool_ext_clear(&ble_hci_acl_pool);
|
|
|
|
if (ret) {
|
|
return ESP_FAIL;
|
|
} else {
|
|
return ESP_OK;
|
|
}
|
|
}
|
|
|
|
esp_err_t esp_nimble_hci_deinit(void)
|
|
{
|
|
if (vhci_send_sem) {
|
|
/* Dummy take & give semaphore before deleting */
|
|
xSemaphoreTake(vhci_send_sem, portMAX_DELAY);
|
|
xSemaphoreGive(vhci_send_sem);
|
|
vSemaphoreDelete(vhci_send_sem);
|
|
vhci_send_sem = NULL;
|
|
}
|
|
return ble_hci_transport_deinit();
|
|
}
|
|
|
|
esp_err_t esp_nimble_hci_and_controller_deinit(void)
|
|
{
|
|
int ret;
|
|
ret = esp_nimble_hci_deinit();
|
|
if (ret != ESP_OK) {
|
|
return ret;
|
|
}
|
|
|
|
ret = esp_bt_controller_disable();
|
|
if (ret != ESP_OK) {
|
|
return ret;
|
|
}
|
|
|
|
ret = esp_bt_controller_deinit();
|
|
if (ret != ESP_OK) {
|
|
return ret;
|
|
}
|
|
|
|
return ESP_OK;
|
|
}
|