1 /*
2  * SPDX-FileCopyrightText: 2015-2024 Espressif Systems (Shanghai) CO LTD
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #include <assert.h>
8 #include "sysinit/sysinit.h"
9 #include "nimble/hci_common.h"
10 #if CONFIG_BT_NIMBLE_ENABLED
11 #include "host/ble_hs.h"
12 #endif //CONFIG_BT_NIMBLE_ENABLED
13 #include "nimble/nimble_port.h"
14 #include "nimble/nimble_port_freertos.h"
15 #include "esp_nimble_hci.h"
16 #include "esp_nimble_mem.h"
17 #include "bt_osi_mem.h"
18 #if CONFIG_BT_CONTROLLER_ENABLED
19 #include "esp_bt.h"
20 #endif
21 #include "freertos/semphr.h"
22 #include "esp_compiler.h"
23 #include "soc/soc_caps.h"
24 #include "bt_common.h"
25 #include "hci_log/bt_hci_log.h"
26 
27 #define NIMBLE_VHCI_TIMEOUT_MS  2000
28 #define BLE_HCI_EVENT_HDR_LEN               (2)
29 #define BLE_HCI_CMD_HDR_LEN                 (3)
30 
31 static ble_hci_trans_rx_cmd_fn *ble_hci_rx_cmd_hs_cb;
32 static void *ble_hci_rx_cmd_hs_arg;
33 
34 static ble_hci_trans_rx_acl_fn *ble_hci_rx_acl_hs_cb;
35 static void *ble_hci_rx_acl_hs_arg;
36 
37 /*
38  * The MBUF payload size must accommodate the HCI data header size plus the
39  * maximum ACL data packet length. The ACL block size is the size of the
40  * mbufs we will allocate.
41  */
42 #define ACL_BLOCK_SIZE  OS_ALIGN(MYNEWT_VAL(BLE_TRANSPORT_ACL_SIZE) \
43         + BLE_MBUF_MEMBLOCK_OVERHEAD \
44         + BLE_HCI_DATA_HDR_SZ, OS_ALIGNMENT)
45 
46 
47 
48 static SemaphoreHandle_t vhci_send_sem;
49 const static char *TAG = "NimBLE";
50 
51 int os_msys_buf_alloc(void);
52 void os_msys_buf_free(void);
53 extern uint8_t ble_hs_enabled_state;
54 
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)55 void ble_hci_trans_cfg_hs(ble_hci_trans_rx_cmd_fn *cmd_cb,
56                           void *cmd_arg,
57                           ble_hci_trans_rx_acl_fn *acl_cb,
58                           void *acl_arg)
59 {
60     ble_hci_rx_cmd_hs_cb = cmd_cb;
61     ble_hci_rx_cmd_hs_arg = cmd_arg;
62     ble_hci_rx_acl_hs_cb = acl_cb;
63     ble_hci_rx_acl_hs_arg = acl_arg;
64 }
65 
esp_vhci_host_send_packet_wrapper(uint8_t * data,uint16_t len)66 void esp_vhci_host_send_packet_wrapper(uint8_t *data, uint16_t len)
67 {
68 #if (BT_HCI_LOG_INCLUDED == TRUE)
69     bt_hci_log_record_hci_data(data[0], &data[1], len - 1);
70 #endif
71     esp_vhci_host_send_packet(data, len);
72 }
73 
ble_hci_trans_hs_cmd_tx(uint8_t * cmd)74 int ble_hci_trans_hs_cmd_tx(uint8_t *cmd)
75 {
76     uint16_t len;
77     uint8_t rc = 0;
78 
79     assert(cmd != NULL);
80     *cmd = BLE_HCI_UART_H4_CMD;
81     len = BLE_HCI_CMD_HDR_LEN + cmd[3] + 1;
82     if (!esp_vhci_host_check_send_available()) {
83         ESP_LOGD(TAG, "Controller not ready to receive packets");
84     }
85 
86     if (xSemaphoreTake(vhci_send_sem, NIMBLE_VHCI_TIMEOUT_MS / portTICK_PERIOD_MS) == pdTRUE) {
87         esp_vhci_host_send_packet_wrapper(cmd, len);
88     } else {
89         rc = BLE_HS_ETIMEOUT_HCI;
90     }
91 
92     ble_transport_free(cmd);
93     return rc;
94 }
95 
ble_hci_trans_ll_evt_tx(uint8_t * hci_ev)96 int ble_hci_trans_ll_evt_tx(uint8_t *hci_ev)
97 {
98     int rc = ESP_FAIL;
99 
100     rc = ble_transport_to_hs_evt((void *)hci_ev);
101     return rc;
102 }
103 
ble_hci_trans_hs_acl_tx(struct os_mbuf * om)104 int ble_hci_trans_hs_acl_tx(struct os_mbuf *om)
105 {
106     uint16_t len = 0;
107     uint8_t data[MYNEWT_VAL(BLE_TRANSPORT_ACL_SIZE) + 1], rc = 0;
108     /* If this packet is zero length, just free it */
109     if (OS_MBUF_PKTLEN(om) == 0) {
110         os_mbuf_free_chain(om);
111         return 0;
112     }
113     data[0] = BLE_HCI_UART_H4_ACL;
114     len++;
115 
116     if (!esp_vhci_host_check_send_available()) {
117         ESP_LOGD(TAG, "Controller not ready to receive packets");
118     }
119 
120     os_mbuf_copydata(om, 0, OS_MBUF_PKTLEN(om), &data[1]);
121     len += OS_MBUF_PKTLEN(om);
122 
123     if (xSemaphoreTake(vhci_send_sem, NIMBLE_VHCI_TIMEOUT_MS / portTICK_PERIOD_MS) == pdTRUE) {
124         esp_vhci_host_send_packet_wrapper(data, len);
125     } else {
126         rc = BLE_HS_ETIMEOUT_HCI;
127     }
128 
129     os_mbuf_free_chain(om);
130 
131     return rc;
132 }
133 
ble_hci_trans_ll_acl_tx(struct os_mbuf * om)134 int ble_hci_trans_ll_acl_tx(struct os_mbuf *om)
135 {
136     int rc = ESP_FAIL;
137 
138     rc = ble_transport_to_hs_acl((void *)om);
139     return rc;
140 }
141 
142 
143 
ble_hci_trans_reset(void)144 int ble_hci_trans_reset(void)
145 {
146     /* No work to do.  All allocated buffers are owned by the host or
147      * controller, and they will get freed by their owners.
148      */
149     return 0;
150 }
151 
152 
ble_hci_rx_acl(uint8_t * data,uint16_t len)153 static void ble_hci_rx_acl(uint8_t *data, uint16_t len)
154 {
155     struct os_mbuf *m = NULL;
156     int rc;
157     int sr;
158     if (len < BLE_HCI_DATA_HDR_SZ || len > MYNEWT_VAL(BLE_TRANSPORT_ACL_SIZE)) {
159         return;
160     }
161 
162     do {
163         m = ble_transport_alloc_acl_from_ll();
164 
165         if (!m) {
166             esp_rom_printf("Failed to allocate buffer, retrying ");
167 	    /* Give some time to free buffer and try again */
168 	    vTaskDelay(1);
169 	}
170     }while(!m);
171 
172     if ((rc = os_mbuf_append(m, data, len)) != 0) {
173         esp_rom_printf("%s failed to os_mbuf_append; rc = %d", __func__, rc);
174         os_mbuf_free_chain(m);
175         return;
176     }
177     OS_ENTER_CRITICAL(sr);
178     ble_transport_to_hs_acl(m);
179     OS_EXIT_CRITICAL(sr);
180 }
181 
182 /*
183  * @brief: BT controller callback function, used to notify the upper layer that
184  *         controller is ready to receive command
185  */
controller_rcv_pkt_ready(void)186 static void controller_rcv_pkt_ready(void)
187 {
188     if (vhci_send_sem) {
189         xSemaphoreGive(vhci_send_sem);
190     }
191 }
192 
dummy_controller_rcv_pkt_ready(void)193 static void dummy_controller_rcv_pkt_ready(void)
194 {
195   /* Dummy function */
196 }
197 
bt_record_hci_data(uint8_t * data,uint16_t len)198 void bt_record_hci_data(uint8_t *data, uint16_t len)
199 {
200 #if (BT_HCI_LOG_INCLUDED == TRUE)
201     if ((data[0] == BLE_HCI_UART_H4_EVT) && (data[1] == BLE_HCI_EVCODE_LE_META) && ((data[3] ==  BLE_HCI_LE_SUBEV_ADV_RPT) || (data[3] == BLE_HCI_LE_SUBEV_DIRECT_ADV_RPT)
202         || (data[3] == BLE_HCI_LE_SUBEV_EXT_ADV_RPT) || (data[3] == BLE_HCI_LE_SUBEV_PERIODIC_ADV_RPT))) {
203         bt_hci_log_record_hci_adv(HCI_LOG_DATA_TYPE_ADV, &data[2], len - 2);
204     } else {
205         uint8_t data_type = ((data[0] == 2) ? HCI_LOG_DATA_TYPE_C2H_ACL : data[0]);
206         bt_hci_log_record_hci_data(data_type, &data[1], len - 1);
207     }
208 #endif // (BT_HCI_LOG_INCLUDED == TRUE)
209 }
210 
dummy_host_rcv_pkt(uint8_t * data,uint16_t len)211 static int dummy_host_rcv_pkt(uint8_t *data, uint16_t len)
212 {
213     /* Dummy function */
214     return 0;
215 }
216 
217 /*
218  * @brief: BT controller callback function, to transfer data packet to the host
219  */
host_rcv_pkt(uint8_t * data,uint16_t len)220 static int host_rcv_pkt(uint8_t *data, uint16_t len)
221 {
222     bt_record_hci_data(data, len);
223 
224     if(!ble_hs_enabled_state) {
225         /* If host is not enabled, drop the packet */
226         esp_rom_printf("Host not enabled. Dropping the packet!");
227         return 0;
228     }
229 
230     if (data[0] == BLE_HCI_UART_H4_EVT) {
231         uint8_t *evbuf;
232         int totlen;
233         int rc;
234 
235         totlen = BLE_HCI_EVENT_HDR_LEN + data[2];
236         assert(totlen <= UINT8_MAX + BLE_HCI_EVENT_HDR_LEN);
237 
238         if (totlen > MYNEWT_VAL(BLE_TRANSPORT_EVT_SIZE)) {
239             esp_rom_printf("Received HCI data length at host (%d) exceeds maximum configured HCI event buffer size (%d).",
240                      totlen, MYNEWT_VAL(BLE_TRANSPORT_EVT_SIZE));
241             ble_hs_sched_reset(BLE_HS_ECONTROLLER);
242             return 0;
243         }
244 
245         if (data[1] == BLE_HCI_EVCODE_HW_ERROR) {
246             assert(0);
247         }
248 
249         /* Allocate LE Advertising Report Event from lo pool only */
250         if ((data[1] == BLE_HCI_EVCODE_LE_META) &&
251                 (data[3] == BLE_HCI_LE_SUBEV_ADV_RPT || data[3] == BLE_HCI_LE_SUBEV_EXT_ADV_RPT)) {
252             evbuf = ble_transport_alloc_evt(1);
253             /* Skip advertising report if we're out of memory */
254             if (!evbuf) {
255                 return 0;
256             }
257         } else {
258             evbuf = ble_transport_alloc_evt(0);
259             assert(evbuf != NULL);
260         }
261 
262         memset(evbuf, 0, sizeof *evbuf);
263         memcpy(evbuf, &data[1], totlen);
264 
265         rc = ble_hci_trans_ll_evt_tx(evbuf);
266         assert(rc == 0);
267     } else if (data[0] == BLE_HCI_UART_H4_ACL) {
268         ble_hci_rx_acl(data + 1, len - 1);
269     }
270     return 0;
271 }
272 
273 static const esp_vhci_host_callback_t vhci_host_cb = {
274     .notify_host_send_available = controller_rcv_pkt_ready,
275     .notify_host_recv = host_rcv_pkt,
276 };
277 
278 static const esp_vhci_host_callback_t dummy_vhci_host_cb = {
279     .notify_host_send_available = dummy_controller_rcv_pkt_ready,
280     .notify_host_recv = dummy_host_rcv_pkt,
281 };
282 
283 
284 extern void ble_transport_init(void);
285 extern esp_err_t ble_buf_alloc(void);
286 extern void ble_buf_free(void);
esp_nimble_hci_init(void)287 esp_err_t esp_nimble_hci_init(void)
288 {
289     esp_err_t ret;
290 
291     ret = ble_buf_alloc();
292     if (ret != ESP_OK) {
293         goto err;
294     }
295     if ((ret = esp_vhci_host_register_callback(&vhci_host_cb)) != ESP_OK) {
296         goto err;
297     }
298 
299     ble_transport_init();
300 
301     vhci_send_sem = xSemaphoreCreateBinary();
302     if (vhci_send_sem == NULL) {
303         ret = ESP_ERR_NO_MEM;
304         goto err;
305     }
306 
307     xSemaphoreGive(vhci_send_sem);
308 
309 #if MYNEWT_VAL(BLE_QUEUE_CONG_CHECK)
310     ble_adv_list_init();
311 #endif
312 
313     return ret;
314 err:
315     ble_buf_free();
316     return ret;
317 
318 }
319 
320 extern void ble_transport_deinit(void);
esp_nimble_hci_deinit(void)321 esp_err_t esp_nimble_hci_deinit(void)
322 {
323     if (vhci_send_sem) {
324         /* Dummy take & give semaphore before deleting */
325         xSemaphoreTake(vhci_send_sem, portMAX_DELAY);
326         xSemaphoreGive(vhci_send_sem);
327         vSemaphoreDelete(vhci_send_sem);
328         vhci_send_sem = NULL;
329     }
330     ble_transport_deinit();
331 
332     esp_vhci_host_register_callback(&dummy_vhci_host_cb);
333 
334     ble_buf_free();
335 
336 #if MYNEWT_VAL(BLE_QUEUE_CONG_CHECK)
337     ble_adv_list_deinit();
338 #endif
339 
340     return ESP_OK;
341 }
342