/* * Copyright (c) 2024 Xiaomi Corporation * Copyright (c) 2015-2016 Intel Corporation * * SPDX-License-Identifier: Apache-2.0 */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define LOG_MODULE_NAME hci_uart_3wire LOG_MODULE_REGISTER(LOG_MODULE_NAME); static K_KERNEL_STACK_DEFINE(tx_stack, CONFIG_BT_HCI_TX_STACK_SIZE); static K_KERNEL_STACK_DEFINE(rx_stack, CONFIG_BT_RX_STACK_SIZE); static struct k_thread tx_thread_data; static struct k_thread rx_thread_data; static struct k_work_delayable ack_work; static struct k_work_delayable retx_work; #define HCI_3WIRE_ACK_PKT 0x00 #define HCI_COMMAND_PKT 0x01 #define HCI_ACLDATA_PKT 0x02 #define HCI_SCODATA_PKT 0x03 #define HCI_EVENT_PKT 0x04 #define HCI_ISODATA_PKT 0x05 #define HCI_3WIRE_LINK_PKT 0x0f #define HCI_VENDOR_PKT 0xff static bool reliable_packet(uint8_t type) { switch (type) { case HCI_COMMAND_PKT: case HCI_ACLDATA_PKT: case HCI_EVENT_PKT: case HCI_ISODATA_PKT: return true; default: return false; } } /* FIXME: Correct timeout */ #define H5_RX_ACK_TIMEOUT K_MSEC(250) #define H5_TX_ACK_TIMEOUT K_MSEC(250) #define SLIP_DELIMITER 0xc0 #define SLIP_ESC 0xdb #define SLIP_ESC_DELIM 0xdc #define SLIP_ESC_ESC 0xdd #define H5_RX_ESC 1 #define H5_TX_ACK_PEND 2 #define H5_HDR_SEQ(hdr) ((hdr)[0] & 0x07) #define H5_HDR_ACK(hdr) (((hdr)[0] >> 3) & 0x07) #define H5_HDR_CRC(hdr) (((hdr)[0] >> 6) & 0x01) #define H5_HDR_RELIABLE(hdr) (((hdr)[0] >> 7) & 0x01) #define H5_HDR_PKT_TYPE(hdr) ((hdr)[1] & 0x0f) #define H5_HDR_LEN(hdr) ((((hdr)[1] >> 4) & 0x0f) + ((hdr)[2] << 4)) #define H5_SET_SEQ(hdr, seq) ((hdr)[0] |= (seq)) #define H5_SET_ACK(hdr, ack) ((hdr)[0] |= (ack) << 3) #define H5_SET_RELIABLE(hdr) ((hdr)[0] |= 1 << 7) #define H5_SET_TYPE(hdr, type) ((hdr)[1] |= type) #define H5_SET_LEN(hdr, len) (((hdr)[1] |= ((len) & 0x0f) << 4), \ ((hdr)[2] |= (len) >> 4)) #define H5_TX_WIN 4 static struct h5 { struct net_buf *rx_buf; struct k_fifo tx_queue; struct k_fifo rx_queue; struct k_fifo unack_queue; uint8_t tx_win; uint8_t tx_ack; uint8_t tx_seq; uint8_t rx_ack; enum { UNINIT, INIT, ACTIVE, } link_state; enum { START, HEADER, PAYLOAD, END, } rx_state; } h5; static uint8_t unack_queue_len; static const uint8_t sync_req[] = { 0x01, 0x7e }; static const uint8_t sync_rsp[] = { 0x02, 0x7d }; /* Third byte may change */ static const uint8_t conf_req[] = { 0x03, 0xfc }; static uint8_t conf_rsp[3] = { 0x04, 0x7b,}; /* H5 signal buffers pool */ #define MAX_SIG_LEN 3 #define SIGNAL_COUNT 2 #define SIG_BUF_SIZE (BT_BUF_RESERVE + MAX_SIG_LEN) NET_BUF_POOL_DEFINE(h5_pool, SIGNAL_COUNT, SIG_BUF_SIZE, 0, NULL); static const struct device *const h5_dev = DEVICE_DT_GET(DT_CHOSEN(zephyr_bt_c2h_uart)); static K_FIFO_DEFINE(tx_queue); static struct k_poll_signal tx_queue_change = K_POLL_SIGNAL_INITIALIZER(tx_queue_change); static void h5_reset_rx(void) { if (h5.rx_buf) { net_buf_unref(h5.rx_buf); h5.rx_buf = NULL; } h5.rx_state = START; } static int h5_unslip_byte(uint8_t *byte) { int count; if (*byte != SLIP_ESC) { return 0; } do { count = uart_fifo_read(h5_dev, byte, sizeof(*byte)); } while (!count); switch (*byte) { case SLIP_ESC_DELIM: *byte = SLIP_DELIMITER; break; case SLIP_ESC_ESC: *byte = SLIP_ESC; break; default: LOG_ERR("Invalid escape byte %x\n", *byte); return -EIO; } return 0; } static void process_unack(void) { uint8_t next_seq = h5.tx_seq; uint8_t number_removed = unack_queue_len; bool acked = false; if (!unack_queue_len) { return; } LOG_DBG("rx_ack %u tx_ack %u tx_seq %u unack_queue_len %u", h5.rx_ack, h5.tx_ack, h5.tx_seq, unack_queue_len); while (unack_queue_len > 0) { if (next_seq == h5.rx_ack) { /* Next sequence number is the same as last received * ack number */ break; } number_removed--; /* Similar to (n - 1) % 8 with unsigned conversion */ next_seq = (next_seq - 1) & 0x07; } if (next_seq != h5.rx_ack) { LOG_ERR("Wrong sequence: rx_ack %u tx_seq %u next_seq %u", h5.rx_ack, h5.tx_seq, next_seq); } LOG_DBG("Need to remove %u packet from the queue", number_removed); while (number_removed) { struct net_buf *buf = k_fifo_get(&h5.unack_queue, K_NO_WAIT); if (!buf) { LOG_ERR("Unack queue is empty"); break; } /* TODO: print or do something with packet */ LOG_DBG("Remove buf from the unack_queue"); net_buf_unref(buf); unack_queue_len--; number_removed--; acked = true; } if (acked) { k_poll_signal_raise(&tx_queue_change, 0); } } static void h5_print_header(const uint8_t *hdr, const char *str) { if (H5_HDR_RELIABLE(hdr)) { LOG_DBG("%s REL: seq %u ack %u crc %u type %u len %u", str, H5_HDR_SEQ(hdr), H5_HDR_ACK(hdr), H5_HDR_CRC(hdr), H5_HDR_PKT_TYPE(hdr), H5_HDR_LEN(hdr)); } else { LOG_DBG("%s UNREL: ack %u crc %u type %u len %u", str, H5_HDR_ACK(hdr), H5_HDR_CRC(hdr), H5_HDR_PKT_TYPE(hdr), H5_HDR_LEN(hdr)); } } static void hexdump(const char *str, const uint8_t *packet, size_t length) { int n = 0; if (!length) { printk("%s zero-length signal packet\n", str); return; } while (length--) { if (n % 16 == 0) { printk("%s %08X ", str, n); } printk("%02X ", *packet++); n++; if (n % 8 == 0) { if (n % 16 == 0) { printk("\n"); } else { printk(" "); } } } if (n % 16) { printk("\n"); } } static uint8_t h5_slip_byte(uint8_t byte) { switch (byte) { case SLIP_DELIMITER: uart_poll_out(h5_dev, SLIP_ESC); uart_poll_out(h5_dev, SLIP_ESC_DELIM); return 2; case SLIP_ESC: uart_poll_out(h5_dev, SLIP_ESC); uart_poll_out(h5_dev, SLIP_ESC_ESC); return 2; default: uart_poll_out(h5_dev, byte); return 1; } } static void h5_send(const uint8_t *payload, uint8_t type, int len) { uint8_t hdr[4]; int i; hexdump("<= ", payload, len); (void)memset(hdr, 0, sizeof(hdr)); /* Set ACK for outgoing packet and stop delayed work */ H5_SET_ACK(hdr, h5.tx_ack); /* If cancel fails we may ack the same seq number twice, this is OK. */ (void)k_work_cancel_delayable(&ack_work); if (reliable_packet(type)) { H5_SET_RELIABLE(hdr); H5_SET_SEQ(hdr, h5.tx_seq); h5.tx_seq = (h5.tx_seq + 1) % 8; } H5_SET_TYPE(hdr, type); H5_SET_LEN(hdr, len); /* Calculate CRC */ hdr[3] = ~((hdr[0] + hdr[1] + hdr[2]) & 0xff); h5_print_header(hdr, "TX: <"); uart_poll_out(h5_dev, SLIP_DELIMITER); for (i = 0; i < 4; i++) { h5_slip_byte(hdr[i]); } for (i = 0; i < len; i++) { h5_slip_byte(payload[i]); } uart_poll_out(h5_dev, SLIP_DELIMITER); } /* Delayed work taking care about retransmitting packets */ static void retx_timeout(struct k_work *work) { ARG_UNUSED(work); LOG_DBG("unack_queue_len %u", unack_queue_len); if (unack_queue_len) { struct k_fifo tmp_queue; struct net_buf *buf; k_fifo_init(&tmp_queue); /* Queue to temporary queue */ while ((buf = k_fifo_get(&h5.tx_queue, K_NO_WAIT))) { k_fifo_put(&tmp_queue, buf); } /* Queue unack packets to the beginning of the queue */ while ((buf = k_fifo_get(&h5.unack_queue, K_NO_WAIT))) { /* include also packet type */ net_buf_push(buf, sizeof(uint8_t)); k_fifo_put(&h5.tx_queue, buf); h5.tx_seq = (h5.tx_seq - 1) & 0x07; unack_queue_len--; } /* Queue saved packets from temp queue */ while ((buf = k_fifo_get(&tmp_queue, K_NO_WAIT))) { k_fifo_put(&h5.tx_queue, buf); } } k_poll_signal_raise(&tx_queue_change, 0); } static void ack_timeout(struct k_work *work) { ARG_UNUSED(work); LOG_DBG(""); h5_send(NULL, HCI_3WIRE_ACK_PKT, 0); } static void h5_process_complete_packet(uint8_t *hdr) { struct net_buf *buf; LOG_DBG(""); /* rx_ack should be in every packet */ h5.rx_ack = H5_HDR_ACK(hdr); if (reliable_packet(H5_HDR_PKT_TYPE(hdr))) { /* For reliable packet increment next transmit ack number */ h5.tx_ack = (h5.tx_ack + 1) % 8; /* Submit delayed work to ack the packet */ k_work_reschedule(&ack_work, H5_RX_ACK_TIMEOUT); } h5_print_header(hdr, "RX: >"); process_unack(); buf = h5.rx_buf; h5.rx_buf = NULL; switch (H5_HDR_PKT_TYPE(hdr)) { case HCI_3WIRE_ACK_PKT: net_buf_unref(buf); break; case HCI_3WIRE_LINK_PKT: k_fifo_put(&h5.rx_queue, buf); break; case HCI_COMMAND_PKT: case HCI_ACLDATA_PKT: case HCI_ISODATA_PKT: hexdump("=> ", buf->data, buf->len); k_fifo_put(&tx_queue, buf); break; } } static void bt_uart_isr(const struct device *unused, void *user_data) { static int remaining; uint8_t byte, type; int ret; static uint8_t hdr[4]; size_t buf_tailroom; ARG_UNUSED(unused); ARG_UNUSED(user_data); while (uart_irq_update(h5_dev) && uart_irq_is_pending(h5_dev)) { if (!uart_irq_rx_ready(h5_dev)) { if (uart_irq_tx_ready(h5_dev)) { LOG_DBG("transmit ready"); } else { LOG_DBG("spurious interrupt"); } /* Only the UART RX path is interrupt-enabled */ break; } ret = uart_fifo_read(h5_dev, &byte, sizeof(byte)); if (!ret) { continue; } switch (h5.rx_state) { case START: if (byte == SLIP_DELIMITER) { h5.rx_state = HEADER; remaining = sizeof(hdr); } break; case HEADER: /* In a case we confuse ending slip delimiter * with starting one. */ if (byte == SLIP_DELIMITER) { remaining = sizeof(hdr); continue; } if (h5_unslip_byte(&byte) < 0) { h5_reset_rx(); continue; } memcpy(&hdr[sizeof(hdr) - remaining], &byte, 1); remaining--; if (remaining) { break; } remaining = H5_HDR_LEN(hdr); type = H5_HDR_PKT_TYPE(hdr); switch (type) { case HCI_COMMAND_PKT: case HCI_ACLDATA_PKT: case HCI_ISODATA_PKT: h5.rx_buf = bt_buf_get_tx(BT_BUF_H4, K_NO_WAIT, &type, sizeof(type)); if (!h5.rx_buf) { LOG_WRN("No available data buffers"); h5_reset_rx(); continue; } h5.rx_state = PAYLOAD; break; case HCI_3WIRE_LINK_PKT: case HCI_3WIRE_ACK_PKT: h5.rx_buf = net_buf_alloc(&h5_pool, K_NO_WAIT); if (!h5.rx_buf) { LOG_WRN("No available signal buffers"); h5_reset_rx(); continue; } h5.rx_state = PAYLOAD; break; default: LOG_ERR("Wrong packet type %u", type); h5.rx_state = END; break; } if (!remaining) { h5.rx_state = END; } break; case PAYLOAD: if (byte == SLIP_DELIMITER) { LOG_WRN("Unexpected ending delimiter"); h5_reset_rx(); continue; } if (h5_unslip_byte(&byte) < 0) { h5_reset_rx(); continue; } buf_tailroom = net_buf_tailroom(h5.rx_buf); if (buf_tailroom < sizeof(byte)) { LOG_ERR("Not enough space in buffer %zu/%zu", sizeof(byte), buf_tailroom); h5_reset_rx(); break; } net_buf_add_mem(h5.rx_buf, &byte, sizeof(byte)); remaining--; if (!remaining) { h5.rx_state = END; } break; case END: if (byte != SLIP_DELIMITER) { LOG_ERR("Missing ending SLIP_DELIMITER"); h5_reset_rx(); break; } LOG_DBG("Received full packet: type %u", H5_HDR_PKT_TYPE(hdr)); /* Check when full packet is received, it can be done * when parsing packet header but we need to receive * full packet anyway to clear UART. */ if (H5_HDR_RELIABLE(hdr) && H5_HDR_SEQ(hdr) != h5.tx_ack) { LOG_ERR("Seq expected %u got %u. Drop packet", h5.tx_ack, H5_HDR_SEQ(hdr)); h5_reset_rx(); break; } h5_process_complete_packet(hdr); h5.rx_state = START; break; } } } static int h5_queue(struct net_buf *buf) { LOG_DBG("buf %p type %u len %u", buf, bt_buf_get_type(buf), buf->len); k_fifo_put(&h5.tx_queue, buf); return 0; } static uint8_t h5_get_type(struct net_buf *buf) { return net_buf_pull_u8(buf); } static void process_events(struct k_poll_event *ev, int count) { struct net_buf *buf; uint8_t type; int err; LOG_DBG("count %d", count); for (; count; ev++, count--) { LOG_DBG("ev->state %u", ev->state); switch (ev->state) { case K_POLL_STATE_SIGNALED: break; case K_POLL_STATE_SEM_AVAILABLE: /* After this fn is exec'd, `bt_conn_prepare_events()` * will be called once again, and this time buffers will * be available, so the FIFO will be added to the poll * list instead of the ctlr buffers semaphore. */ break; case K_POLL_STATE_FIFO_DATA_AVAILABLE: if (ev->tag == 0) { /* Wait until a buffer is available */ buf = k_fifo_get(&tx_queue, K_NO_WAIT); __ASSERT_NO_MSG(buf); /* Pass buffer to the stack */ err = bt_send(buf); if (err) { LOG_ERR("Unable to send (err %d)", err); net_buf_unref(buf); } } else if (ev->tag == 2) { buf = k_fifo_get(&h5.tx_queue, K_FOREVER); __ASSERT_NO_MSG(buf); type = h5_get_type(buf); h5_send(buf->data, type, buf->len); /* buf is dequeued from tx_queue and queued to unack * queue. */ k_fifo_put(&h5.unack_queue, buf); unack_queue_len++; k_work_reschedule(&retx_work, H5_TX_ACK_TIMEOUT); } break; case K_POLL_STATE_NOT_READY: break; default: LOG_WRN("Unexpected k_poll event state %u", ev->state); break; } } } static void tx_thread(void *p1, void *p2, void *p3) { static struct k_poll_event events[] = { K_POLL_EVENT_STATIC_INITIALIZER(K_POLL_TYPE_FIFO_DATA_AVAILABLE, K_POLL_MODE_NOTIFY_ONLY, &tx_queue, 0), K_POLL_EVENT_STATIC_INITIALIZER(K_POLL_TYPE_SIGNAL, K_POLL_MODE_NOTIFY_ONLY, &tx_queue_change, 1), K_POLL_EVENT_STATIC_INITIALIZER(K_POLL_TYPE_FIFO_DATA_AVAILABLE, K_POLL_MODE_NOTIFY_ONLY, &h5.tx_queue, 2), }; ARG_UNUSED(p1); ARG_UNUSED(p2); ARG_UNUSED(p3); LOG_DBG(""); while (true) { int err, ev_count = 2; events[0].state = K_POLL_STATE_NOT_READY; events[1].state = K_POLL_STATE_NOT_READY; tx_queue_change.signaled = 0U; if (h5.link_state == ACTIVE && unack_queue_len < h5.tx_win) { events[2].state = K_POLL_STATE_NOT_READY; ev_count++; } err = k_poll(events, ev_count, K_FOREVER); process_events(events, ev_count); /* Make sure we don't hog the CPU if there's all the time * some ready events. */ k_yield(); } } static void rx_thread(void *p1, void *p2, void *p3) { ARG_UNUSED(p1); ARG_UNUSED(p2); ARG_UNUSED(p3); LOG_DBG(""); while (true) { struct net_buf *buf, *cache; buf = k_fifo_get(&h5.rx_queue, K_FOREVER); hexdump("=> ", buf->data, buf->len); if (!memcmp(buf->data, sync_req, sizeof(sync_req))) { if (h5.link_state == ACTIVE) { while ((cache = k_fifo_get(&h5.unack_queue, K_NO_WAIT))) { net_buf_unref(cache); } unack_queue_len = 0; while ((cache = k_fifo_get(&h5.tx_queue, K_NO_WAIT))) { net_buf_unref(cache); } h5_reset_rx(); h5.rx_ack = 0; h5.link_state = INIT; h5.tx_ack = 0; h5.tx_seq = 0; } h5_send(sync_rsp, HCI_3WIRE_LINK_PKT, sizeof(sync_rsp)); } else if (!memcmp(buf->data, conf_req, 2)) { if (buf->len > 2) { uint8_t tx_win = buf->data[2] & 0x07; /* Configuration field present */ h5.tx_win = MIN(h5.tx_win, tx_win); } conf_rsp[2] = h5.tx_win; /* * The Host sends Config Response messages with a * Configuration Field. */ h5_send(conf_rsp, HCI_3WIRE_LINK_PKT, sizeof(conf_rsp)); LOG_DBG("Finished H5 configuration, tx_win %u", h5.tx_win); h5.link_state = ACTIVE; } else { LOG_ERR("Not handled yet %x %x", buf->data[0], buf->data[1]); } net_buf_unref(buf); /* Make sure we don't hog the CPU if the rx_queue never * gets empty. */ k_yield(); } } static int hci_uart_init(void) { LOG_DBG(""); if (!device_is_ready(h5_dev)) { LOG_ERR("HCI UART %s is not ready", h5_dev->name); return -EINVAL; } uart_irq_rx_disable(h5_dev); uart_irq_tx_disable(h5_dev); uart_irq_callback_set(h5_dev, bt_uart_isr); uart_irq_rx_enable(h5_dev); return 0; } SYS_INIT(hci_uart_init, APPLICATION, CONFIG_KERNEL_INIT_PRIORITY_DEVICE); int main(void) { /* incoming events and data from the controller */ static K_FIFO_DEFINE(rx_queue); int err; LOG_DBG("Start"); __ASSERT(h5_dev, "UART device is NULL"); /* Enable the raw interface, this will in turn open the HCI driver */ bt_enable_raw(&rx_queue); /* TX thread */ k_fifo_init(&h5.tx_queue); k_thread_create(&tx_thread_data, tx_stack, K_KERNEL_STACK_SIZEOF(tx_stack), tx_thread, NULL, NULL, NULL, K_PRIO_COOP(CONFIG_BT_HCI_TX_PRIO), 0, K_NO_WAIT); k_thread_name_set(&tx_thread_data, "tx_thread"); k_fifo_init(&h5.rx_queue); k_thread_create(&rx_thread_data, rx_stack, K_KERNEL_STACK_SIZEOF(rx_stack), rx_thread, NULL, NULL, NULL, K_PRIO_COOP(CONFIG_BT_RX_PRIO), 0, K_NO_WAIT); k_thread_name_set(&rx_thread_data, "rx_thread"); /* Unack queue */ k_fifo_init(&h5.unack_queue); /* Init delayed work */ k_work_init_delayable(&ack_work, ack_timeout); k_work_init_delayable(&retx_work, retx_timeout); h5.tx_win = H5_TX_WIN; while (1) { struct net_buf *buf; buf = k_fifo_get(&rx_queue, K_FOREVER); err = h5_queue(buf); if (err) { LOG_ERR("Failed to send"); } } return 0; }