/* * Copyright (c) 2021-2025 Nordic Semiconductor ASA * * SPDX-License-Identifier: Apache-2.0 */ #include #include #include #include #include #include #include #include #include #include #include #include #include static const struct bt_data ad[] = { BT_DATA_BYTES(BT_DATA_FLAGS, (BT_LE_AD_GENERAL | BT_LE_AD_NO_BREDR)), }; static const struct bt_data sd[] = { BT_DATA(BT_DATA_NAME_COMPLETE, CONFIG_BT_DEVICE_NAME, sizeof(CONFIG_BT_DEVICE_NAME) - 1), }; static void connected(struct bt_conn *conn, uint8_t err) { char addr[BT_ADDR_LE_STR_LEN]; bt_addr_le_to_str(bt_conn_get_dst(conn), addr, sizeof(addr)); if (err) { printk("Failed to connect to %s %u %s\n", addr, err, bt_hci_err_to_str(err)); return; } printk("Connected %s\n", addr); } static void disconnected(struct bt_conn *conn, uint8_t reason) { char addr[BT_ADDR_LE_STR_LEN]; bt_addr_le_to_str(bt_conn_get_dst(conn), addr, sizeof(addr)); printk("Disconnected from %s, reason 0x%02x %s\n", addr, reason, bt_hci_err_to_str(reason)); } BT_CONN_CB_DEFINE(conn_callbacks) = { .connected = connected, .disconnected = disconnected, }; /** Print data as d_0 d_1 d_2 ... d_(n-2) d_(n-1) d_(n) to show the 3 first and 3 last octets * * Examples: * 01 * 0102 * 010203 * 01020304 * 0102030405 * 010203040506 * 010203...050607 * 010203...060708 * etc. */ static void iso_print_data(uint8_t *data, size_t data_len) { /* Maximum number of octets from each end of the data */ const uint8_t max_octets = 3; char data_str[35]; size_t str_len; str_len = bin2hex(data, MIN(max_octets, data_len), data_str, sizeof(data_str)); if (data_len > max_octets) { if (data_len > (max_octets * 2)) { static const char dots[] = "..."; strcat(&data_str[str_len], dots); str_len += strlen(dots); } str_len += bin2hex(data + (data_len - MIN(max_octets, data_len - max_octets)), MIN(max_octets, data_len - max_octets), data_str + str_len, sizeof(data_str) - str_len); } printk("\t %s\n", data_str); } static void iso_recv(struct bt_iso_chan *chan, const struct bt_iso_recv_info *info, struct net_buf *buf) { if (info->flags & BT_ISO_FLAGS_VALID) { printk("Incoming data channel %p len %u\n", chan, buf->len); iso_print_data(buf->data, buf->len); } } static void iso_connected(struct bt_iso_chan *chan) { const struct bt_iso_chan_path hci_path = { .pid = BT_ISO_DATA_PATH_HCI, .format = BT_HCI_CODING_FORMAT_TRANSPARENT, }; int err; printk("ISO Channel %p connected\n", chan); err = bt_iso_setup_data_path(chan, BT_HCI_DATAPATH_DIR_CTLR_TO_HOST, &hci_path); if (err != 0) { printk("Failed to setup ISO RX data path: %d", err); } } static void iso_disconnected(struct bt_iso_chan *chan, uint8_t reason) { printk("ISO Channel %p disconnected (reason 0x%02x)\n", chan, reason); } static struct bt_iso_chan_ops iso_ops = { .recv = iso_recv, .connected = iso_connected, .disconnected = iso_disconnected, }; static struct bt_iso_chan_io_qos iso_rx = { .sdu = CONFIG_BT_ISO_TX_MTU, }; static struct bt_iso_chan_qos iso_qos = { .rx = &iso_rx, .tx = NULL, }; static struct bt_iso_chan iso_chan = { .ops = &iso_ops, .qos = &iso_qos, }; static int iso_accept(const struct bt_iso_accept_info *info, struct bt_iso_chan **chan) { printk("Incoming request from %p\n", (void *)info->acl); if (iso_chan.iso) { printk("No channels available\n"); return -ENOMEM; } *chan = &iso_chan; return 0; } static struct bt_iso_server iso_server = { #if defined(CONFIG_BT_SMP) .sec_level = BT_SECURITY_L1, #endif /* CONFIG_BT_SMP */ .accept = iso_accept, }; int main(void) { int err; err = bt_enable(NULL); if (err) { printk("Bluetooth init failed (err %d)\n", err); return 0; } if (IS_ENABLED(CONFIG_SETTINGS)) { settings_load(); } printk("Bluetooth initialized\n"); err = bt_iso_server_register(&iso_server); if (err) { printk("Unable to register ISO server (err %d)\n", err); return 0; } err = bt_le_adv_start(BT_LE_ADV_CONN_FAST_1, ad, ARRAY_SIZE(ad), sd, ARRAY_SIZE(sd)); if (err) { printk("Advertising failed to start (err %d)\n", err); return 0; } printk("Advertising successfully started\n"); return 0; }