Files
CherryUSB/third_party/nimble-1.6.0/ble_hci_usbh.c
2024-01-22 19:49:23 +08:00

152 lines
3.1 KiB
C

/*
* Copyright (c) 2024, sakumisu
*
* SPDX-License-Identifier: Apache-2.0
*/
#include "usbh_core.h"
#include "usbh_bluetooth.h"
#include <nimble/ble.h>
#include "nimble/transport.h"
#include "nimble/transport/hci_h4.h"
struct hci_h4_sm g_hci_h4sm;
void ble_transport_ll_init(void)
{
/* nothing here */
}
static void hci_dump(uint8_t hci_type, uint8_t *data, uint32_t len)
{
uint32_t i = 0;
USB_LOG_DBG("\r\nhci type:%u", hci_type);
for (i = 0; i < len; i++) {
if (i % 16 == 0) {
USB_LOG_DBG("\r\n");
}
USB_LOG_DBG("%02x ", data[i]);
}
USB_LOG_DBG("\r\n");
}
static int hci_usb_frame_cb(uint8_t pkt_type, void *data)
{
switch (pkt_type) {
case HCI_H4_EVT:
return ble_transport_to_hs_evt(data);
case HCI_H4_ACL:
return ble_transport_to_hs_acl(data);
default:
assert(0);
break;
}
return -1;
}
static int ble_usb_transport_init(void)
{
hci_h4_sm_init(&g_hci_h4sm, &hci_h4_allocs_from_ll, hci_usb_frame_cb);
return 0;
}
void usbh_bluetooth_hci_rx_callback(uint8_t hci_type, uint8_t *data, uint32_t len)
{
uint8_t pkt_type = 0;
switch (hci_type) {
case USB_BLUETOOTH_HCI_EVT:
pkt_type = HCI_H4_EVT;
break;
case USB_BLUETOOTH_HCI_ACL_IN:
pkt_type = HCI_H4_ACL;
break;
// case USB_BLUETOOTH_HCI_ISO_IN:
// break;
default:
USB_LOG_ERR("Unknown HCI type %u\r\n", hci_type);
return;
}
hci_dump(pkt_type, data, len);
hci_h4_sm_rx(&g_hci_h4sm, &pkt_type, 1);
hci_h4_sm_rx(&g_hci_h4sm, data, len);
}
int ble_transport_to_ll_cmd_impl(void *buf)
{
int ret = 0;
uint8_t *cmd_pkt_data = (uint8_t *)buf;
size_t pkt_len = cmd_pkt_data[2] + 3;
hci_dump(HCI_H4_CMD, buf, pkt_len);
ret = usbh_bluetooth_hci_cmd(buf, pkt_len);
if (ret < 0) {
ret = BLE_ERR_MEM_CAPACITY;
} else {
ret = 0;
}
ble_transport_free(buf);
return ret;
}
int ble_transport_to_ll_acl_impl(struct os_mbuf *om)
{
struct os_mbuf *x = om;
int ret = 0;
while (x != NULL) {
hci_dump(HCI_H4_ACL, x->om_data, x->om_len);
ret = usbh_bluetooth_hci_acl_out(x->om_data, x->om_len);
if (ret < 0) {
ret = BLE_ERR_MEM_CAPACITY;
break;
} else {
ret = 0;
}
x = SLIST_NEXT(x, om_next);
}
os_mbuf_free_chain(om);
return ret;
}
__WEAK void usbh_bluetooth_run_callback(void)
{
/* bt_enable() */
}
__WEAK void usbh_bluetooth_stop_callback(void)
{
/* bt_disable() */
}
void usbh_bluetooth_run(struct usbh_bluetooth *bluetooth_class)
{
ble_usb_transport_init();
usb_osal_thread_create("ble_event", 2048, CONFIG_USBHOST_PSC_PRIO + 1, usbh_bluetooth_hci_event_rx_thread, NULL);
usb_osal_thread_create("ble_acl", 2048, CONFIG_USBHOST_PSC_PRIO + 1, usbh_bluetooth_hci_acl_rx_thread, NULL);
usbh_bluetooth_run_callback();
}
void usbh_bluetooth_stop(struct usbh_bluetooth *bluetooth_class)
{
usbh_bluetooth_stop_callback();
}