|  | /** @file | 
|  | * @brief Bluetooth shell module | 
|  | * | 
|  | * Provide some Bluetooth shell commands that can be useful to applications. | 
|  | */ | 
|  |  | 
|  | /* | 
|  | * Copyright (c) 2017 Intel Corporation | 
|  | * Copyright (c) 2018 Nordic Semiconductor ASA | 
|  | * | 
|  | * SPDX-License-Identifier: Apache-2.0 | 
|  | */ | 
|  |  | 
|  | #include <errno.h> | 
|  | #include <zephyr/types.h> | 
|  | #include <stddef.h> | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  | #include <sys/printk.h> | 
|  | #include <sys/byteorder.h> | 
|  | #include <sys/util.h> | 
|  | #include <zephyr.h> | 
|  |  | 
|  | #include <settings/settings.h> | 
|  |  | 
|  | #include <bluetooth/hci.h> | 
|  | #include <bluetooth/bluetooth.h> | 
|  | #include <bluetooth/conn.h> | 
|  | #include <bluetooth/rfcomm.h> | 
|  | #include <bluetooth/sdp.h> | 
|  | #include <bluetooth/iso.h> | 
|  |  | 
|  | #include <shell/shell.h> | 
|  |  | 
|  | #include "bt.h" | 
|  | #include "ll.h" | 
|  | #include "hci.h" | 
|  |  | 
|  | #define DEVICE_NAME		CONFIG_BT_DEVICE_NAME | 
|  | #define DEVICE_NAME_LEN		(sizeof(DEVICE_NAME) - 1) | 
|  |  | 
|  | /* Multiply bt 1.25 to get MS */ | 
|  | #define BT_INTERVAL_TO_MS(interval) ((interval) * 5 / 4) | 
|  |  | 
|  | static bool no_settings_load; | 
|  |  | 
|  | uint8_t selected_id = BT_ID_DEFAULT; | 
|  | const struct shell *ctx_shell; | 
|  |  | 
|  | #if defined(CONFIG_BT_CONN) | 
|  | struct bt_conn *default_conn; | 
|  |  | 
|  | /* Connection context for BR/EDR legacy pairing in sec mode 3 */ | 
|  | static struct bt_conn *pairing_conn; | 
|  |  | 
|  | static struct bt_le_oob oob_local; | 
|  | #if defined(CONFIG_BT_SMP) || defined(CONFIG_BT_BREDR) | 
|  | static struct bt_le_oob oob_remote; | 
|  | #endif /* CONFIG_BT_SMP || CONFIG_BT_BREDR) */ | 
|  | #endif /* CONFIG_BT_CONN */ | 
|  |  | 
|  | #define NAME_LEN 30 | 
|  |  | 
|  | #define KEY_STR_LEN 33 | 
|  |  | 
|  | /* | 
|  | * Based on the maximum number of parameters for HCI_LE_Generate_DHKey | 
|  | * See BT Core Spec V5.2 Vol. 4, Part E, section 7.8.37 | 
|  | */ | 
|  | #define HCI_CMD_MAX_PARAM 65 | 
|  |  | 
|  | #if defined(CONFIG_BT_EXT_ADV) | 
|  | uint8_t selected_adv; | 
|  | struct bt_le_ext_adv *adv_sets[CONFIG_BT_EXT_ADV_MAX_ADV_SET]; | 
|  | #endif /* CONFIG_BT_EXT_ADV */ | 
|  |  | 
|  | #if defined(CONFIG_BT_OBSERVER) || defined(CONFIG_BT_USER_PHY_UPDATE) | 
|  | static const char *phy2str(uint8_t phy) | 
|  | { | 
|  | switch (phy) { | 
|  | case 0: return "No packets"; | 
|  | case BT_GAP_LE_PHY_1M: return "LE 1M"; | 
|  | case BT_GAP_LE_PHY_2M: return "LE 2M"; | 
|  | case BT_GAP_LE_PHY_CODED: return "LE Coded"; | 
|  | default: return "Unknown"; | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if defined(CONFIG_BT_OBSERVER) | 
|  | static bool data_cb(struct bt_data *data, void *user_data) | 
|  | { | 
|  | char *name = user_data; | 
|  |  | 
|  | switch (data->type) { | 
|  | case BT_DATA_NAME_SHORTENED: | 
|  | case BT_DATA_NAME_COMPLETE: | 
|  | memcpy(name, data->data, MIN(data->data_len, NAME_LEN - 1)); | 
|  | return false; | 
|  | default: | 
|  | return true; | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | static void scan_recv(const struct bt_le_scan_recv_info *info, | 
|  | struct net_buf_simple *buf) | 
|  | { | 
|  | char le_addr[BT_ADDR_LE_STR_LEN]; | 
|  | char name[NAME_LEN]; | 
|  |  | 
|  | (void)memset(name, 0, sizeof(name)); | 
|  |  | 
|  | bt_data_parse(buf, data_cb, name); | 
|  |  | 
|  | bt_addr_le_to_str(info->addr, le_addr, sizeof(le_addr)); | 
|  | shell_print(ctx_shell, "[DEVICE]: %s, AD evt type %u, RSSI %i %s " | 
|  | "C:%u S:%u D:%d SR:%u E:%u Prim: %s, Secn: %s, " | 
|  | "Interval: 0x%04x (%u ms), SID: 0x%x", | 
|  | le_addr, info->adv_type, info->rssi, name, | 
|  | (info->adv_props & BT_GAP_ADV_PROP_CONNECTABLE) != 0, | 
|  | (info->adv_props & BT_GAP_ADV_PROP_SCANNABLE) != 0, | 
|  | (info->adv_props & BT_GAP_ADV_PROP_DIRECTED) != 0, | 
|  | (info->adv_props & BT_GAP_ADV_PROP_SCAN_RESPONSE) != 0, | 
|  | (info->adv_props & BT_GAP_ADV_PROP_EXT_ADV) != 0, | 
|  | phy2str(info->primary_phy), phy2str(info->secondary_phy), | 
|  | info->interval, BT_INTERVAL_TO_MS(info->interval), | 
|  | info->sid); | 
|  | } | 
|  |  | 
|  | static void scan_timeout(void) | 
|  | { | 
|  | shell_print(ctx_shell, "Scan timeout"); | 
|  | } | 
|  | #endif /* CONFIG_BT_OBSERVER */ | 
|  |  | 
|  | #if defined(CONFIG_BT_EXT_ADV) | 
|  | #if defined(CONFIG_BT_BROADCASTER) | 
|  | static void adv_sent(struct bt_le_ext_adv *adv, | 
|  | struct bt_le_ext_adv_sent_info *info) | 
|  | { | 
|  | shell_print(ctx_shell, "Advertiser[%d] %p sent %d", | 
|  | bt_le_ext_adv_get_index(adv), adv, info->num_sent); | 
|  | } | 
|  |  | 
|  | static void adv_scanned(struct bt_le_ext_adv *adv, | 
|  | struct bt_le_ext_adv_scanned_info *info) | 
|  | { | 
|  | char str[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(info->addr, str, sizeof(str)); | 
|  |  | 
|  | shell_print(ctx_shell, "Advertiser[%d] %p scanned by %s", | 
|  | bt_le_ext_adv_get_index(adv), adv, str); | 
|  | } | 
|  | #endif /* CONFIG_BT_BROADCASTER */ | 
|  |  | 
|  | #if defined(CONFIG_BT_PERIPHERAL) | 
|  | static void adv_connected(struct bt_le_ext_adv *adv, | 
|  | struct bt_le_ext_adv_connected_info *info) | 
|  | { | 
|  | char str[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(bt_conn_get_dst(info->conn), str, sizeof(str)); | 
|  |  | 
|  | shell_print(ctx_shell, "Advertiser[%d] %p connected by %s", | 
|  | bt_le_ext_adv_get_index(adv), adv, str); | 
|  | } | 
|  | #endif /* CONFIG_BT_PERIPHERAL */ | 
|  | #endif /* CONFIG_BT_EXT_ADV */ | 
|  |  | 
|  | #if !defined(CONFIG_BT_CONN) | 
|  | #if 0 /* FIXME: Add support for changing prompt */ | 
|  | static const char *current_prompt(void) | 
|  | { | 
|  | return NULL; | 
|  | } | 
|  | #endif | 
|  | #endif /* !CONFIG_BT_CONN */ | 
|  |  | 
|  | #if defined(CONFIG_BT_CONN) | 
|  | #if 0 /* FIXME: Add support for changing prompt */ | 
|  | static const char *current_prompt(void) | 
|  | { | 
|  | static char str[BT_ADDR_LE_STR_LEN + 2]; | 
|  | static struct bt_conn_info info; | 
|  |  | 
|  | if (!default_conn) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | if (bt_conn_get_info(default_conn, &info) < 0) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | if (info.type != BT_CONN_TYPE_LE) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | bt_addr_le_to_str(info.le.dst, str, sizeof(str) - 2); | 
|  | strcat(str, "> "); | 
|  | return str; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | void conn_addr_str(struct bt_conn *conn, char *addr, size_t len) | 
|  | { | 
|  | struct bt_conn_info info; | 
|  |  | 
|  | if (bt_conn_get_info(conn, &info) < 0) { | 
|  | addr[0] = '\0'; | 
|  | return; | 
|  | } | 
|  |  | 
|  | switch (info.type) { | 
|  | #if defined(CONFIG_BT_BREDR) | 
|  | case BT_CONN_TYPE_BR: | 
|  | bt_addr_to_str(info.br.dst, addr, len); | 
|  | break; | 
|  | #endif | 
|  | case BT_CONN_TYPE_LE: | 
|  | bt_addr_le_to_str(info.le.dst, addr, len); | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | static void print_le_oob(const struct shell *shell, struct bt_le_oob *oob) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  | char c[KEY_STR_LEN]; | 
|  | char r[KEY_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(&oob->addr, addr, sizeof(addr)); | 
|  |  | 
|  | bin2hex(oob->le_sc_data.c, sizeof(oob->le_sc_data.c), c, sizeof(c)); | 
|  | bin2hex(oob->le_sc_data.r, sizeof(oob->le_sc_data.r), r, sizeof(r)); | 
|  |  | 
|  | shell_print(shell, "OOB data:"); | 
|  | shell_print(shell, "%-29s %-32s %-32s", "addr", "random", "confirm"); | 
|  | shell_print(shell, "%29s %32s %32s", addr, r, c); | 
|  | } | 
|  |  | 
|  | static void connected(struct bt_conn *conn, uint8_t err) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | conn_addr_str(conn, addr, sizeof(addr)); | 
|  |  | 
|  | if (err) { | 
|  | shell_error(ctx_shell, "Failed to connect to %s (0x%02x)", addr, | 
|  | err); | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | shell_print(ctx_shell, "Connected: %s", addr); | 
|  |  | 
|  | if (!default_conn) { | 
|  | default_conn = bt_conn_ref(conn); | 
|  | } | 
|  |  | 
|  | done: | 
|  | /* clear connection reference for sec mode 3 pairing */ | 
|  | if (pairing_conn) { | 
|  | bt_conn_unref(pairing_conn); | 
|  | pairing_conn = NULL; | 
|  | } | 
|  | } | 
|  |  | 
|  | static void disconnected(struct bt_conn *conn, uint8_t reason) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | conn_addr_str(conn, addr, sizeof(addr)); | 
|  | shell_print(ctx_shell, "Disconnected: %s (reason 0x%02x)", addr, reason); | 
|  |  | 
|  | if (default_conn == conn) { | 
|  | bt_conn_unref(default_conn); | 
|  | default_conn = NULL; | 
|  | } | 
|  | } | 
|  |  | 
|  | static bool le_param_req(struct bt_conn *conn, struct bt_le_conn_param *param) | 
|  | { | 
|  | shell_print(ctx_shell, "LE conn  param req: int (0x%04x, 0x%04x) lat %d" | 
|  | " to %d", param->interval_min, param->interval_max, | 
|  | param->latency, param->timeout); | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static void le_param_updated(struct bt_conn *conn, uint16_t interval, | 
|  | uint16_t latency, uint16_t timeout) | 
|  | { | 
|  | shell_print(ctx_shell, "LE conn param updated: int 0x%04x lat %d " | 
|  | "to %d", interval, latency, timeout); | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_SMP) | 
|  | static void identity_resolved(struct bt_conn *conn, const bt_addr_le_t *rpa, | 
|  | const bt_addr_le_t *identity) | 
|  | { | 
|  | char addr_identity[BT_ADDR_LE_STR_LEN]; | 
|  | char addr_rpa[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(identity, addr_identity, sizeof(addr_identity)); | 
|  | bt_addr_le_to_str(rpa, addr_rpa, sizeof(addr_rpa)); | 
|  |  | 
|  | shell_print(ctx_shell, "Identity resolved %s -> %s", addr_rpa, | 
|  | addr_identity); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if defined(CONFIG_BT_SMP) || defined(CONFIG_BT_BREDR) | 
|  | static const char *security_err_str(enum bt_security_err err) | 
|  | { | 
|  | switch (err) { | 
|  | case BT_SECURITY_ERR_SUCCESS: | 
|  | return "Success"; | 
|  | case BT_SECURITY_ERR_AUTH_FAIL: | 
|  | return "Authentication failure"; | 
|  | case BT_SECURITY_ERR_PIN_OR_KEY_MISSING: | 
|  | return "PIN or key missing"; | 
|  | case BT_SECURITY_ERR_OOB_NOT_AVAILABLE: | 
|  | return "OOB not available"; | 
|  | case BT_SECURITY_ERR_AUTH_REQUIREMENT: | 
|  | return "Authentication requirements"; | 
|  | case BT_SECURITY_ERR_PAIR_NOT_SUPPORTED: | 
|  | return "Pairing not supported"; | 
|  | case BT_SECURITY_ERR_PAIR_NOT_ALLOWED: | 
|  | return "Pairing not allowed"; | 
|  | case BT_SECURITY_ERR_INVALID_PARAM: | 
|  | return "Invalid parameters"; | 
|  | case BT_SECURITY_ERR_UNSPECIFIED: | 
|  | return "Unspecified"; | 
|  | default: | 
|  | return "Unknown"; | 
|  | } | 
|  | } | 
|  |  | 
|  | static void security_changed(struct bt_conn *conn, bt_security_t level, | 
|  | enum bt_security_err err) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | conn_addr_str(conn, addr, sizeof(addr)); | 
|  |  | 
|  | if (!err) { | 
|  | shell_print(ctx_shell, "Security changed: %s level %u", addr, | 
|  | level); | 
|  | } else { | 
|  | shell_print(ctx_shell, "Security failed: %s level %u " | 
|  | "reason: %s (%d)", | 
|  | addr, level, security_err_str(err), err); | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if defined(CONFIG_BT_REMOTE_INFO) | 
|  | static const char *ver_str(uint8_t ver) | 
|  | { | 
|  | const char * const str[] = { | 
|  | "1.0b", "1.1", "1.2", "2.0", "2.1", "3.0", "4.0", "4.1", "4.2", | 
|  | "5.0", "5.1", | 
|  | }; | 
|  |  | 
|  | if (ver < ARRAY_SIZE(str)) { | 
|  | return str[ver]; | 
|  | } | 
|  |  | 
|  | return "unknown"; | 
|  | } | 
|  |  | 
|  | static void remote_info_available(struct bt_conn *conn, | 
|  | struct bt_conn_remote_info *remote_info) | 
|  | { | 
|  | struct bt_conn_info info; | 
|  |  | 
|  | bt_conn_get_info(conn, &info); | 
|  |  | 
|  | if (IS_ENABLED(CONFIG_BT_REMOTE_VERSION)) { | 
|  | shell_print(ctx_shell, | 
|  | "Remote LMP version %s (0x%02x) subversion 0x%04x " | 
|  | "manufacturer 0x%04x", ver_str(remote_info->version), | 
|  | remote_info->version, remote_info->subversion, | 
|  | remote_info->manufacturer); | 
|  | } | 
|  |  | 
|  | if (info.type == BT_CONN_TYPE_LE) { | 
|  | uint8_t features[8]; | 
|  | char features_str[2 * sizeof(features) +  1]; | 
|  |  | 
|  | sys_memcpy_swap(features, remote_info->le.features, | 
|  | sizeof(features)); | 
|  | bin2hex(features, sizeof(features), | 
|  | features_str, sizeof(features_str)); | 
|  | shell_print(ctx_shell, "LE Features: 0x%s ", features_str); | 
|  | } | 
|  | } | 
|  | #endif /* defined(CONFIG_BT_REMOTE_INFO) */ | 
|  |  | 
|  | #if defined(CONFIG_BT_USER_DATA_LEN_UPDATE) | 
|  | void le_data_len_updated(struct bt_conn *conn, | 
|  | struct bt_conn_le_data_len_info *info) | 
|  | { | 
|  | shell_print(ctx_shell, | 
|  | "LE data len updated: TX (len: %d time: %d)" | 
|  | " RX (len: %d time: %d)", info->tx_max_len, | 
|  | info->tx_max_time, info->rx_max_len, info->rx_max_time); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if defined(CONFIG_BT_USER_PHY_UPDATE) | 
|  | void le_phy_updated(struct bt_conn *conn, | 
|  | struct bt_conn_le_phy_info *info) | 
|  | { | 
|  | shell_print(ctx_shell, "LE PHY updated: TX PHY %s, RX PHY %s", | 
|  | phy2str(info->tx_phy), phy2str(info->rx_phy)); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | static struct bt_conn_cb conn_callbacks = { | 
|  | .connected = connected, | 
|  | .disconnected = disconnected, | 
|  | .le_param_req = le_param_req, | 
|  | .le_param_updated = le_param_updated, | 
|  | #if defined(CONFIG_BT_SMP) | 
|  | .identity_resolved = identity_resolved, | 
|  | #endif | 
|  | #if defined(CONFIG_BT_SMP) || defined(CONFIG_BT_BREDR) | 
|  | .security_changed = security_changed, | 
|  | #endif | 
|  | #if defined(CONFIG_BT_REMOTE_INFO) | 
|  | .remote_info_available = remote_info_available, | 
|  | #endif | 
|  | #if defined(CONFIG_BT_USER_DATA_LEN_UPDATE) | 
|  | .le_data_len_updated = le_data_len_updated, | 
|  | #endif | 
|  | #if defined(CONFIG_BT_USER_PHY_UPDATE) | 
|  | .le_phy_updated = le_phy_updated, | 
|  | #endif | 
|  | }; | 
|  | #endif /* CONFIG_BT_CONN */ | 
|  |  | 
|  | #if defined(CONFIG_BT_OBSERVER) | 
|  | static struct bt_le_scan_cb scan_callbacks = { | 
|  | .recv = scan_recv, | 
|  | .timeout = scan_timeout, | 
|  | }; | 
|  | #endif /* defined(CONFIG_BT_OBSERVER) */ | 
|  |  | 
|  | #if defined(CONFIG_BT_EXT_ADV) | 
|  | #if defined(CONFIG_BT_BROADCASTER) | 
|  | static struct bt_le_ext_adv_cb adv_callbacks = { | 
|  | .sent = adv_sent, | 
|  | .scanned = adv_scanned, | 
|  | #if defined(CONFIG_BT_PERIPHERAL) | 
|  | .connected = adv_connected, | 
|  | #endif /* CONFIG_BT_PERIPHERAL */ | 
|  | }; | 
|  | #endif /* CONFIG_BT_BROADCASTER */ | 
|  | #endif /* CONFIG_BT_EXT_ADV */ | 
|  |  | 
|  |  | 
|  | #if defined(CONFIG_BT_PER_ADV_SYNC) | 
|  | struct bt_le_per_adv_sync *per_adv_syncs[CONFIG_BT_PER_ADV_SYNC_MAX]; | 
|  |  | 
|  | static void per_adv_sync_sync_cb(struct bt_le_per_adv_sync *sync, | 
|  | struct bt_le_per_adv_sync_synced_info *info) | 
|  | { | 
|  | char le_addr[BT_ADDR_LE_STR_LEN]; | 
|  | char past_peer[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(info->addr, le_addr, sizeof(le_addr)); | 
|  |  | 
|  | if (info->conn) { | 
|  | conn_addr_str(info->conn, past_peer, sizeof(past_peer)); | 
|  | } else { | 
|  | memset(past_peer, 0, sizeof(past_peer)); | 
|  | } | 
|  |  | 
|  | shell_print(ctx_shell, "PER_ADV_SYNC[%u]: [DEVICE]: %s synced, " | 
|  | "Interval 0x%04x (%u ms), PHY %s, SD 0x%04X, PAST peer %s", | 
|  | bt_le_per_adv_sync_get_index(sync), le_addr, | 
|  | info->interval, BT_INTERVAL_TO_MS(info->interval), | 
|  | phy2str(info->phy), info->service_data, past_peer); | 
|  |  | 
|  | if (info->conn) { /* if from PAST */ | 
|  | for (int i = 0; i < ARRAY_SIZE(per_adv_syncs); i++) { | 
|  | if (!per_adv_syncs[i]) { | 
|  | per_adv_syncs[i] = sync; | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | static void per_adv_sync_terminated_cb( | 
|  | struct bt_le_per_adv_sync *sync, | 
|  | const struct bt_le_per_adv_sync_term_info *info) | 
|  | { | 
|  | char le_addr[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | for (int i = 0; i < ARRAY_SIZE(per_adv_syncs); i++) { | 
|  | if (per_adv_syncs[i] == sync) { | 
|  | per_adv_syncs[i] = NULL; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | bt_addr_le_to_str(info->addr, le_addr, sizeof(le_addr)); | 
|  | shell_print(ctx_shell, "PER_ADV_SYNC[%u]: [DEVICE]: %s sync terminated", | 
|  | bt_le_per_adv_sync_get_index(sync), le_addr); | 
|  | } | 
|  |  | 
|  | static void per_adv_sync_recv_cb( | 
|  | struct bt_le_per_adv_sync *sync, | 
|  | const struct bt_le_per_adv_sync_recv_info *info, | 
|  | struct net_buf_simple *buf) | 
|  | { | 
|  | char le_addr[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(info->addr, le_addr, sizeof(le_addr)); | 
|  | shell_print(ctx_shell, "PER_ADV_SYNC[%u]: [DEVICE]: %s, tx_power %i, " | 
|  | "RSSI %i, CTE %u, data length %u", | 
|  | bt_le_per_adv_sync_get_index(sync), le_addr, info->tx_power, | 
|  | info->rssi, info->cte_type, buf->len); | 
|  | } | 
|  |  | 
|  | static void per_adv_sync_biginfo_cb(struct bt_le_per_adv_sync *sync, | 
|  | const struct bt_iso_biginfo *biginfo) | 
|  | { | 
|  | char le_addr[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(biginfo->addr, le_addr, sizeof(le_addr)); | 
|  | shell_print(ctx_shell, "PER_ADV_SYNC[%u]: [DEVICE]: %s, sid 0x%02x, num_bis %u, " | 
|  | "nse 0x%02x, interval 0x%04x (%u ms), bn 0x%02x, pto 0x%02x, irc 0x%02x, " | 
|  | "max_pdu 0x%04x, sdu_interval 0x%04x, max_sdu 0x%04x, phy %s, framing 0x%02x, " | 
|  | "%sencrypted", | 
|  | bt_le_per_adv_sync_get_index(sync), le_addr, biginfo->sid, biginfo->num_bis, | 
|  | biginfo->sub_evt_count, biginfo->iso_interval, | 
|  | BT_INTERVAL_TO_MS(biginfo->iso_interval), biginfo->burst_number, | 
|  | biginfo->offset, biginfo->rep_count, biginfo->max_pdu, biginfo->sdu_interval, | 
|  | biginfo->max_sdu, phy2str(biginfo->phy), biginfo->framing, | 
|  | biginfo->encryption ? "" : "not "); | 
|  | } | 
|  |  | 
|  | static struct bt_le_per_adv_sync_cb per_adv_sync_cb = { | 
|  | .synced = per_adv_sync_sync_cb, | 
|  | .term = per_adv_sync_terminated_cb, | 
|  | .recv = per_adv_sync_recv_cb, | 
|  | .biginfo = per_adv_sync_biginfo_cb, | 
|  | }; | 
|  | #endif /* CONFIG_BT_PER_ADV_SYNC */ | 
|  |  | 
|  | static void bt_ready(int err) | 
|  | { | 
|  | if (err) { | 
|  | shell_error(ctx_shell, "Bluetooth init failed (err %d)", err); | 
|  | return; | 
|  | } | 
|  |  | 
|  | shell_print(ctx_shell, "Bluetooth initialized"); | 
|  |  | 
|  | if (IS_ENABLED(CONFIG_SETTINGS) && !no_settings_load) { | 
|  | settings_load(); | 
|  | shell_print(ctx_shell, "Settings Loaded"); | 
|  | } | 
|  |  | 
|  | if (IS_ENABLED(CONFIG_BT_SMP_OOB_LEGACY_PAIR_ONLY)) { | 
|  | bt_set_oob_data_flag(true); | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_OBSERVER) | 
|  | bt_le_scan_cb_register(&scan_callbacks); | 
|  | #endif | 
|  |  | 
|  | #if defined(CONFIG_BT_CONN) | 
|  | default_conn = NULL; | 
|  |  | 
|  | bt_conn_cb_register(&conn_callbacks); | 
|  | #endif /* CONFIG_BT_CONN */ | 
|  |  | 
|  | #if defined(CONFIG_BT_PER_ADV_SYNC) | 
|  | bt_le_per_adv_sync_cb_register(&per_adv_sync_cb); | 
|  | #endif /* CONFIG_BT_PER_ADV_SYNC */ | 
|  | } | 
|  |  | 
|  | static int cmd_init(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | int err; | 
|  | bool no_ready_cb = false; | 
|  |  | 
|  | ctx_shell = shell; | 
|  |  | 
|  | for (size_t argn = 1; argn < argc; argn++) { | 
|  | const char *arg = argv[argn]; | 
|  |  | 
|  | if (!strcmp(arg, "no-settings-load")) { | 
|  | no_settings_load = true; | 
|  | } else if (!strcmp(arg, "sync")) { | 
|  | no_ready_cb = true; | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (no_ready_cb) { | 
|  | err = bt_enable(bt_ready); | 
|  | if (err) { | 
|  | shell_error(shell, "Bluetooth init failed (err %d)", | 
|  | err); | 
|  | } | 
|  |  | 
|  | } else { | 
|  | err = bt_enable(NULL); | 
|  | bt_ready(err); | 
|  | } | 
|  |  | 
|  | return err; | 
|  | } | 
|  |  | 
|  | #ifdef CONFIG_SETTINGS | 
|  | static int cmd_settings_load(const struct shell *shell, size_t argc, | 
|  | char *argv[]) | 
|  | { | 
|  | int err; | 
|  |  | 
|  | err = settings_load(); | 
|  | if (err) { | 
|  | shell_error(shell, "Settings load failed (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | shell_print(shell, "Settings loaded"); | 
|  | return 0; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if defined(CONFIG_BT_HCI) | 
|  | static int cmd_hci_cmd(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | uint8_t ogf; | 
|  | uint16_t ocf; | 
|  | struct net_buf *buf = NULL, *rsp; | 
|  | int err; | 
|  | static uint8_t hex_data[HCI_CMD_MAX_PARAM]; | 
|  | int hex_data_len; | 
|  |  | 
|  | hex_data_len = 0; | 
|  | ogf = strtoul(argv[1], NULL, 16); | 
|  | ocf = strtoul(argv[2], NULL, 16); | 
|  |  | 
|  | if (argc > 3) { | 
|  | size_t len; | 
|  |  | 
|  | if (strlen(argv[3]) > 2 * HCI_CMD_MAX_PARAM) { | 
|  | shell_error(shell, "Data field too large\n"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | len = hex2bin(argv[3], strlen(argv[3]), &hex_data[hex_data_len], | 
|  | sizeof(hex_data) - hex_data_len); | 
|  | if (!len) { | 
|  | shell_error(shell, "HCI command illegal data field\n"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | buf = bt_hci_cmd_create(BT_OP(ogf, ocf), len); | 
|  | net_buf_add_mem(buf, hex_data, len); | 
|  | } | 
|  |  | 
|  | err = bt_hci_cmd_send_sync(BT_OP(ogf, ocf), buf, &rsp); | 
|  | if (err) { | 
|  | shell_error(shell, "HCI command failed (err %d)", err); | 
|  | return err; | 
|  | } else { | 
|  | shell_hexdump(shell, rsp->data, rsp->len); | 
|  | net_buf_unref(rsp); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | #endif /* CONFIG_BT_HCI */ | 
|  |  | 
|  | static int cmd_name(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | int err; | 
|  |  | 
|  | if (argc < 2) { | 
|  | shell_print(shell, "Bluetooth Local Name: %s", bt_get_name()); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | err = bt_set_name(argv[1]); | 
|  | if (err) { | 
|  | shell_error(shell, "Unable to set name %s (err %d)", argv[1], | 
|  | err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_id_create(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | char addr_str[BT_ADDR_LE_STR_LEN]; | 
|  | bt_addr_le_t addr; | 
|  | int err; | 
|  |  | 
|  | if (argc > 1) { | 
|  | err = bt_addr_le_from_str(argv[1], "random", &addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid address"); | 
|  | } | 
|  | } else { | 
|  | bt_addr_le_copy(&addr, BT_ADDR_LE_ANY); | 
|  | } | 
|  |  | 
|  | err = bt_id_create(&addr, NULL); | 
|  | if (err < 0) { | 
|  | shell_error(shell, "Creating new ID failed (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | bt_addr_le_to_str(&addr, addr_str, sizeof(addr_str)); | 
|  | shell_print(shell, "New identity (%d) created: %s", err, addr_str); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_id_reset(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | char addr_str[BT_ADDR_LE_STR_LEN]; | 
|  | bt_addr_le_t addr; | 
|  | uint8_t id; | 
|  | int err; | 
|  |  | 
|  | if (argc < 2) { | 
|  | shell_error(shell, "Identity identifier not specified"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | id = strtol(argv[1], NULL, 10); | 
|  |  | 
|  | if (argc > 2) { | 
|  | err = bt_addr_le_from_str(argv[2], "random", &addr); | 
|  | if (err) { | 
|  | shell_print(shell, "Invalid address"); | 
|  | return err; | 
|  | } | 
|  | } else { | 
|  | bt_addr_le_copy(&addr, BT_ADDR_LE_ANY); | 
|  | } | 
|  |  | 
|  | err = bt_id_reset(id, &addr, NULL); | 
|  | if (err < 0) { | 
|  | shell_print(shell, "Resetting ID %u failed (err %d)", id, err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | bt_addr_le_to_str(&addr, addr_str, sizeof(addr_str)); | 
|  | shell_print(shell, "Identity %u reset: %s", id, addr_str); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_id_delete(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | uint8_t id; | 
|  | int err; | 
|  |  | 
|  | if (argc < 2) { | 
|  | shell_error(shell, "Identity identifier not specified"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | id = strtol(argv[1], NULL, 10); | 
|  |  | 
|  | err = bt_id_delete(id); | 
|  | if (err < 0) { | 
|  | shell_error(shell, "Deleting ID %u failed (err %d)", id, err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | shell_print(shell, "Identity %u deleted", id); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_id_show(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | bt_addr_le_t addrs[CONFIG_BT_ID_MAX]; | 
|  | size_t i, count = CONFIG_BT_ID_MAX; | 
|  |  | 
|  | bt_id_get(addrs, &count); | 
|  |  | 
|  | for (i = 0; i < count; i++) { | 
|  | char addr_str[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(&addrs[i], addr_str, sizeof(addr_str)); | 
|  | shell_print(shell, "%s%zu: %s", i == selected_id ? "*" : " ", i, | 
|  | addr_str); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_id_select(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | char addr_str[BT_ADDR_LE_STR_LEN]; | 
|  | bt_addr_le_t addrs[CONFIG_BT_ID_MAX]; | 
|  | size_t count = CONFIG_BT_ID_MAX; | 
|  | uint8_t id; | 
|  |  | 
|  | id = strtol(argv[1], NULL, 10); | 
|  |  | 
|  | bt_id_get(addrs, &count); | 
|  | if (count <= id) { | 
|  | shell_error(shell, "Invalid identity"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | bt_addr_le_to_str(&addrs[id], addr_str, sizeof(addr_str)); | 
|  | shell_print(shell, "Selected identity: %s", addr_str); | 
|  | selected_id = id; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_OBSERVER) | 
|  | static int cmd_active_scan_on(const struct shell *shell, uint32_t options, | 
|  | uint16_t timeout) | 
|  | { | 
|  | int err; | 
|  | struct bt_le_scan_param param = { | 
|  | .type       = BT_LE_SCAN_TYPE_ACTIVE, | 
|  | .options    = BT_LE_SCAN_OPT_NONE, | 
|  | .interval   = BT_GAP_SCAN_FAST_INTERVAL, | 
|  | .window     = BT_GAP_SCAN_FAST_WINDOW, | 
|  | .timeout    = timeout, }; | 
|  |  | 
|  | param.options |= options; | 
|  |  | 
|  | err = bt_le_scan_start(¶m, NULL); | 
|  | if (err) { | 
|  | shell_error(shell, "Bluetooth set active scan failed " | 
|  | "(err %d)", err); | 
|  | return err; | 
|  | } else { | 
|  | shell_print(shell, "Bluetooth active scan enabled"); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_passive_scan_on(const struct shell *shell, uint32_t options, | 
|  | uint16_t timeout) | 
|  | { | 
|  | struct bt_le_scan_param param = { | 
|  | .type       = BT_LE_SCAN_TYPE_PASSIVE, | 
|  | .options    = BT_LE_SCAN_OPT_NONE, | 
|  | .interval   = 0x10, | 
|  | .window     = 0x10, | 
|  | .timeout    = timeout, }; | 
|  | int err; | 
|  |  | 
|  | param.options |= options; | 
|  |  | 
|  | err = bt_le_scan_start(¶m, NULL); | 
|  | if (err) { | 
|  | shell_error(shell, "Bluetooth set passive scan failed " | 
|  | "(err %d)", err); | 
|  | return err; | 
|  | } else { | 
|  | shell_print(shell, "Bluetooth passive scan enabled"); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_scan_off(const struct shell *shell) | 
|  | { | 
|  | int err; | 
|  |  | 
|  | err = bt_le_scan_stop(); | 
|  | if (err) { | 
|  | shell_error(shell, "Stopping scanning failed (err %d)", err); | 
|  | return err; | 
|  | } else { | 
|  | shell_print(shell, "Scan successfully stopped"); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_scan(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | const char *action; | 
|  | uint32_t options = 0; | 
|  | uint16_t timeout = 0; | 
|  |  | 
|  | /* Parse duplicate filtering data */ | 
|  | for (size_t argn = 2; argn < argc; argn++) { | 
|  | const char *arg = argv[argn]; | 
|  |  | 
|  | if (!strcmp(arg, "dups")) { | 
|  | options |= BT_LE_SCAN_OPT_FILTER_DUPLICATE; | 
|  | } else if (!strcmp(arg, "nodups")) { | 
|  | options &= ~BT_LE_SCAN_OPT_FILTER_DUPLICATE; | 
|  | } else if (!strcmp(arg, "wl")) { | 
|  | options |= BT_LE_SCAN_OPT_FILTER_WHITELIST; | 
|  | } else if (!strcmp(arg, "coded")) { | 
|  | options |= BT_LE_SCAN_OPT_CODED; | 
|  | } else if (!strcmp(arg, "no-1m")) { | 
|  | options |= BT_LE_SCAN_OPT_NO_1M; | 
|  | } else if (!strcmp(arg, "timeout")) { | 
|  | if (++argn == argc) { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | timeout = strtoul(argv[argn], NULL, 16); | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  | } | 
|  |  | 
|  | action = argv[1]; | 
|  | if (!strcmp(action, "on")) { | 
|  | return cmd_active_scan_on(shell, options, timeout); | 
|  | } else if (!strcmp(action, "off")) { | 
|  | return cmd_scan_off(shell); | 
|  | } else if (!strcmp(action, "passive")) { | 
|  | return cmd_passive_scan_on(shell, options, timeout); | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | #endif /* CONFIG_BT_OBSERVER */ | 
|  |  | 
|  | #if defined(CONFIG_BT_BROADCASTER) | 
|  | static const struct bt_data ad_discov[] = { | 
|  | BT_DATA_BYTES(BT_DATA_FLAGS, (BT_LE_AD_GENERAL | BT_LE_AD_NO_BREDR)), | 
|  | }; | 
|  |  | 
|  | static int cmd_advertise(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_le_adv_param param = {}; | 
|  | const struct bt_data *ad; | 
|  | size_t ad_len; | 
|  | int err; | 
|  |  | 
|  | if (!strcmp(argv[1], "off")) { | 
|  | if (bt_le_adv_stop() < 0) { | 
|  | shell_error(shell, "Failed to stop advertising"); | 
|  | return -ENOEXEC; | 
|  | } else { | 
|  | shell_print(shell, "Advertising stopped"); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | param.id = selected_id; | 
|  | param.interval_min = BT_GAP_ADV_FAST_INT_MIN_2; | 
|  | param.interval_max = BT_GAP_ADV_FAST_INT_MAX_2; | 
|  |  | 
|  | if (!strcmp(argv[1], "on")) { | 
|  | param.options = (BT_LE_ADV_OPT_CONNECTABLE | | 
|  | BT_LE_ADV_OPT_USE_NAME); | 
|  | } else if (!strcmp(argv[1], "scan")) { | 
|  | param.options = BT_LE_ADV_OPT_USE_NAME; | 
|  | } else if (!strcmp(argv[1], "nconn")) { | 
|  | param.options = 0U; | 
|  | } else { | 
|  | goto fail; | 
|  | } | 
|  |  | 
|  | ad = ad_discov; | 
|  | ad_len = ARRAY_SIZE(ad_discov); | 
|  |  | 
|  | for (size_t argn = 2; argn < argc; argn++) { | 
|  | const char *arg = argv[argn]; | 
|  |  | 
|  | if (!strcmp(arg, "discov")) { | 
|  | /* Default */ | 
|  | } else if (!strcmp(arg, "non_discov")) { | 
|  | ad = NULL; | 
|  | ad_len = 0; | 
|  | } else if (!strcmp(arg, "wl")) { | 
|  | param.options |= BT_LE_ADV_OPT_FILTER_SCAN_REQ; | 
|  | param.options |= BT_LE_ADV_OPT_FILTER_CONN; | 
|  | } else if (!strcmp(arg, "wl-scan")) { | 
|  | param.options |= BT_LE_ADV_OPT_FILTER_SCAN_REQ; | 
|  | } else if (!strcmp(arg, "wl-conn")) { | 
|  | param.options |= BT_LE_ADV_OPT_FILTER_CONN; | 
|  | } else if (!strcmp(arg, "identity")) { | 
|  | param.options |= BT_LE_ADV_OPT_USE_IDENTITY; | 
|  | } else if (!strcmp(arg, "no-name")) { | 
|  | param.options &= ~BT_LE_ADV_OPT_USE_NAME; | 
|  | } else if (!strcmp(arg, "one-time")) { | 
|  | param.options |= BT_LE_ADV_OPT_ONE_TIME; | 
|  | } else if (!strcmp(arg, "disable-37")) { | 
|  | param.options |= BT_LE_ADV_OPT_DISABLE_CHAN_37; | 
|  | } else if (!strcmp(arg, "disable-38")) { | 
|  | param.options |= BT_LE_ADV_OPT_DISABLE_CHAN_38; | 
|  | } else if (!strcmp(arg, "disable-39")) { | 
|  | param.options |= BT_LE_ADV_OPT_DISABLE_CHAN_39; | 
|  | } else { | 
|  | goto fail; | 
|  | } | 
|  | } | 
|  |  | 
|  | err = bt_le_adv_start(¶m, ad, ad_len, NULL, 0); | 
|  | if (err < 0) { | 
|  | shell_error(shell, "Failed to start advertising (err %d)", | 
|  | err); | 
|  | return err; | 
|  | } else { | 
|  | shell_print(shell, "Advertising started"); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  |  | 
|  | fail: | 
|  | shell_help(shell); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_PERIPHERAL) | 
|  | static int cmd_directed_adv(const struct shell *shell, | 
|  | size_t argc, char *argv[]) | 
|  | { | 
|  | int err; | 
|  | bt_addr_le_t addr; | 
|  | struct bt_le_adv_param param; | 
|  |  | 
|  | err = bt_addr_le_from_str(argv[1], argv[2], &addr); | 
|  | param = *BT_LE_ADV_CONN_DIR(&addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid peer address (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | for (size_t argn = 3; argn < argc; argn++) { | 
|  | const char *arg = argv[argn]; | 
|  |  | 
|  | if (!strcmp(arg, "low")) { | 
|  | param.options |= BT_LE_ADV_OPT_DIR_MODE_LOW_DUTY; | 
|  | param.interval_max = BT_GAP_ADV_FAST_INT_MAX_2; | 
|  | param.interval_min = BT_GAP_ADV_FAST_INT_MIN_2; | 
|  | } else if (!strcmp(arg, "identity")) { | 
|  | param.options |= BT_LE_ADV_OPT_USE_IDENTITY; | 
|  | } else if (!strcmp(arg, "dir-rpa")) { | 
|  | param.options |= BT_LE_ADV_OPT_DIR_ADDR_RPA; | 
|  | } else if (!strcmp(arg, "disable-37")) { | 
|  | param.options |= BT_LE_ADV_OPT_DISABLE_CHAN_37; | 
|  | } else if (!strcmp(arg, "disable-38")) { | 
|  | param.options |= BT_LE_ADV_OPT_DISABLE_CHAN_38; | 
|  | } else if (!strcmp(arg, "disable-39")) { | 
|  | param.options |= BT_LE_ADV_OPT_DISABLE_CHAN_39; | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return -ENOEXEC; | 
|  | } | 
|  | } | 
|  |  | 
|  | err = bt_le_adv_start(¶m, NULL, 0, NULL, 0); | 
|  | if (err) { | 
|  | shell_error(shell, "Failed to start directed advertising (%d)", | 
|  | err); | 
|  | return -ENOEXEC; | 
|  | } else { | 
|  | shell_print(shell, "Started directed advertising"); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | #endif /* CONFIG_BT_PERIPHERAL */ | 
|  |  | 
|  | #if defined(CONFIG_BT_EXT_ADV) | 
|  | static bool adv_param_parse(size_t argc, char *argv[], | 
|  | struct bt_le_adv_param *param) | 
|  | { | 
|  | memset(param, 0, sizeof(struct bt_le_adv_param)); | 
|  |  | 
|  | if (!strcmp(argv[1], "conn-scan")) { | 
|  | param->options |= BT_LE_ADV_OPT_CONNECTABLE; | 
|  | param->options |= BT_LE_ADV_OPT_SCANNABLE; | 
|  | } else if (!strcmp(argv[1], "conn-nscan")) { | 
|  | param->options |= BT_LE_ADV_OPT_CONNECTABLE; | 
|  | } else if (!strcmp(argv[1], "nconn-scan")) { | 
|  | param->options |= BT_LE_ADV_OPT_SCANNABLE; | 
|  | } else if (!strcmp(argv[1], "nconn-nscan")) { | 
|  | /* Acceptable option, nothing to do */ | 
|  | } else { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | for (size_t argn = 2; argn < argc; argn++) { | 
|  | const char *arg = argv[argn]; | 
|  |  | 
|  | if (!strcmp(arg, "ext-adv")) { | 
|  | param->options |= BT_LE_ADV_OPT_EXT_ADV; | 
|  | } else if (!strcmp(arg, "coded")) { | 
|  | param->options |= BT_LE_ADV_OPT_CODED; | 
|  | } else if (!strcmp(arg, "no-2m")) { | 
|  | param->options |= BT_LE_ADV_OPT_NO_2M; | 
|  | } else if (!strcmp(arg, "anon")) { | 
|  | param->options |= BT_LE_ADV_OPT_ANONYMOUS; | 
|  | } else if (!strcmp(arg, "tx-power")) { | 
|  | param->options |= BT_LE_ADV_OPT_USE_TX_POWER; | 
|  | } else if (!strcmp(arg, "scan-reports")) { | 
|  | param->options |= BT_LE_ADV_OPT_NOTIFY_SCAN_REQ; | 
|  | } else if (!strcmp(arg, "wl")) { | 
|  | param->options |= BT_LE_ADV_OPT_FILTER_SCAN_REQ; | 
|  | param->options |= BT_LE_ADV_OPT_FILTER_CONN; | 
|  | } else if (!strcmp(arg, "wl-scan")) { | 
|  | param->options |= BT_LE_ADV_OPT_FILTER_SCAN_REQ; | 
|  | } else if (!strcmp(arg, "wl-conn")) { | 
|  | param->options |= BT_LE_ADV_OPT_FILTER_CONN; | 
|  | } else if (!strcmp(arg, "identity")) { | 
|  | param->options |= BT_LE_ADV_OPT_USE_IDENTITY; | 
|  | } else if (!strcmp(arg, "name")) { | 
|  | param->options |= BT_LE_ADV_OPT_USE_NAME; | 
|  | } else if (!strcmp(arg, "low")) { | 
|  | param->options |= BT_LE_ADV_OPT_DIR_MODE_LOW_DUTY; | 
|  | } else if (!strcmp(arg, "disable-37")) { | 
|  | param->options |= BT_LE_ADV_OPT_DISABLE_CHAN_37; | 
|  | } else if (!strcmp(arg, "disable-38")) { | 
|  | param->options |= BT_LE_ADV_OPT_DISABLE_CHAN_38; | 
|  | } else if (!strcmp(arg, "disable-39")) { | 
|  | param->options |= BT_LE_ADV_OPT_DISABLE_CHAN_39; | 
|  | } else if (!strcmp(arg, "directed")) { | 
|  | static bt_addr_le_t addr; | 
|  |  | 
|  | if ((argn + 2) >= argc) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | if (bt_addr_le_from_str(argv[argn + 1], argv[argn + 2], | 
|  | &addr)) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | param->peer = &addr; | 
|  | argn += 2; | 
|  | } else { | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | param->id = selected_id; | 
|  | param->sid = 0; | 
|  | if (param->peer && | 
|  | !(param->options & BT_LE_ADV_OPT_DIR_MODE_LOW_DUTY)) { | 
|  | param->interval_min = 0; | 
|  | param->interval_max = 0; | 
|  | } else { | 
|  | param->interval_min = BT_GAP_ADV_FAST_INT_MIN_2; | 
|  | param->interval_max = BT_GAP_ADV_FAST_INT_MAX_2; | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static int cmd_adv_create(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_le_adv_param param; | 
|  | struct bt_le_ext_adv *adv; | 
|  | uint8_t adv_index; | 
|  | int err; | 
|  |  | 
|  | if (!adv_param_parse(argc, argv, ¶m)) { | 
|  | shell_help(shell); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | err = bt_le_ext_adv_create(¶m, &adv_callbacks, &adv); | 
|  | if (err) { | 
|  | shell_error(shell, "Failed to create advertiser set (%d)", err); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | adv_index = bt_le_ext_adv_get_index(adv); | 
|  | adv_sets[adv_index] = adv; | 
|  |  | 
|  | shell_print(shell, "Created adv id: %d, adv: %p", adv_index, adv); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_adv_param(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_le_ext_adv *adv = adv_sets[selected_adv]; | 
|  | struct bt_le_adv_param param; | 
|  | int err; | 
|  |  | 
|  | if (!adv_param_parse(argc, argv, ¶m)) { | 
|  | shell_help(shell); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | err = bt_le_ext_adv_update_param(adv, ¶m); | 
|  | if (err) { | 
|  | shell_error(shell, "Failed to update advertiser set (%d)", err); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_adv_data(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | uint8_t discov_data = (BT_LE_AD_GENERAL | BT_LE_AD_NO_BREDR); | 
|  | struct bt_le_ext_adv *adv = adv_sets[selected_adv]; | 
|  | static uint8_t hex_data[1650]; | 
|  | struct bt_data *data; | 
|  | struct bt_data ad[8]; | 
|  | struct bt_data sd[8]; | 
|  | size_t hex_data_len; | 
|  | size_t ad_len = 0; | 
|  | size_t sd_len = 0; | 
|  | size_t *data_len; | 
|  | int err; | 
|  |  | 
|  | if (!adv) { | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | hex_data_len = 0; | 
|  | data = ad; | 
|  | data_len = &ad_len; | 
|  |  | 
|  | for (size_t argn = 1; argn < argc; argn++) { | 
|  | const char *arg = argv[argn]; | 
|  |  | 
|  | if (strcmp(arg, "scan-response") && | 
|  | *data_len == ARRAY_SIZE(ad)) { | 
|  | /* Maximum entries limit reached. */ | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | if (!strcmp(arg, "discov")) { | 
|  | data[*data_len].type = BT_DATA_FLAGS; | 
|  | data[*data_len].data_len = sizeof(discov_data); | 
|  | data[*data_len].data = &discov_data; | 
|  | (*data_len)++; | 
|  | } else if (!strcmp(arg, "name")) { | 
|  | const char *name = bt_get_name(); | 
|  |  | 
|  | data[*data_len].type = BT_DATA_NAME_COMPLETE; | 
|  | data[*data_len].data_len = strlen(name); | 
|  | data[*data_len].data = name; | 
|  | (*data_len)++; | 
|  | } else if (!strcmp(arg, "scan-response")) { | 
|  | if (data == sd) { | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | data = sd; | 
|  | data_len = &sd_len; | 
|  | } else { | 
|  | size_t len; | 
|  |  | 
|  | len = hex2bin(arg, strlen(arg), &hex_data[hex_data_len], | 
|  | sizeof(hex_data) - hex_data_len); | 
|  |  | 
|  | if (!len || (len - 1) != (hex_data[hex_data_len])) { | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | data[*data_len].type = hex_data[hex_data_len + 1]; | 
|  | data[*data_len].data_len = hex_data[hex_data_len]; | 
|  | data[*data_len].data = &hex_data[hex_data_len + 2]; | 
|  | (*data_len)++; | 
|  | hex_data_len += len; | 
|  | } | 
|  | } | 
|  |  | 
|  | err = bt_le_ext_adv_set_data(adv, ad_len > 0 ? ad : NULL, ad_len, | 
|  | sd_len > 0 ? sd : NULL, sd_len); | 
|  | if (err) { | 
|  | shell_print(shell, "Failed to set advertising set data (%d)", | 
|  | err); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_adv_start(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_le_ext_adv *adv = adv_sets[selected_adv]; | 
|  | struct bt_le_ext_adv_start_param param; | 
|  | uint8_t num_events = 0; | 
|  | int32_t timeout = 0; | 
|  | int err; | 
|  |  | 
|  | if (!adv) { | 
|  | shell_print(shell, "Advertiser[%d] not created", selected_adv); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | for (size_t argn = 1; argn < argc; argn++) { | 
|  | const char *arg = argv[argn]; | 
|  |  | 
|  | if (!strcmp(arg, "timeout")) { | 
|  | if (++argn == argc) { | 
|  | goto fail_show_help; | 
|  | } | 
|  |  | 
|  | timeout = strtoul(argv[argn], NULL, 16); | 
|  | } | 
|  |  | 
|  | if (!strcmp(arg, "num-events")) { | 
|  | if (++argn == argc) { | 
|  | goto fail_show_help; | 
|  | } | 
|  |  | 
|  | num_events = strtoul(argv[argn], NULL, 16); | 
|  | } | 
|  | } | 
|  |  | 
|  | param.timeout = timeout; | 
|  | param.num_events = num_events; | 
|  |  | 
|  | err = bt_le_ext_adv_start(adv, ¶m); | 
|  | if (err) { | 
|  | shell_print(shell, "Failed to start advertising set (%d)", err); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | shell_print(shell, "Advertiser[%d] %p set started", selected_adv, adv); | 
|  | return 0; | 
|  |  | 
|  | fail_show_help: | 
|  | shell_help(shell); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | static int cmd_adv_stop(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_le_ext_adv *adv = adv_sets[selected_adv]; | 
|  | int err; | 
|  |  | 
|  | if (!adv) { | 
|  | shell_print(shell, "Advertiser[%d] not created", selected_adv); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | err = bt_le_ext_adv_stop(adv); | 
|  | if (err) { | 
|  | shell_print(shell, "Failed to stop advertising set (%d)", err); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | shell_print(shell, "Advertiser set stopped"); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_adv_delete(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_le_ext_adv *adv = adv_sets[selected_adv]; | 
|  | int err; | 
|  |  | 
|  | if (!adv) { | 
|  | shell_print(shell, "Advertiser[%d] not created", selected_adv); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | err = bt_le_ext_adv_delete(adv); | 
|  | if (err) { | 
|  | shell_error(ctx_shell, "Failed to delete advertiser set"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | adv_sets[selected_adv] = NULL; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_adv_select(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | if (argc == 2) { | 
|  | uint8_t id = strtol(argv[1], NULL, 10); | 
|  |  | 
|  | if (!(id < ARRAY_SIZE(adv_sets))) { | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | selected_adv = id; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | for (int i = 0; i < ARRAY_SIZE(adv_sets); i++) { | 
|  | if (adv_sets[i]) { | 
|  | shell_print(shell, "Advertiser[%d] %p", i, adv_sets[i]); | 
|  | } | 
|  | } | 
|  |  | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | static int cmd_adv_info(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_le_ext_adv *adv = adv_sets[selected_adv]; | 
|  | struct bt_le_ext_adv_info info; | 
|  | int err; | 
|  |  | 
|  | if (!adv) { | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | err = bt_le_ext_adv_get_info(adv, &info); | 
|  | if (err) { | 
|  | shell_error(shell, "OOB data failed"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | shell_print(shell, "Advertiser[%d] %p", selected_adv, adv); | 
|  | shell_print(shell, "Id: %d, TX power: %d dBm", info.id, info.tx_power); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_PERIPHERAL) | 
|  | static int cmd_adv_oob(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_le_ext_adv *adv = adv_sets[selected_adv]; | 
|  | int err; | 
|  |  | 
|  | if (!adv) { | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | err = bt_le_ext_adv_oob_get_local(adv, &oob_local); | 
|  | if (err) { | 
|  | shell_error(shell, "OOB data failed"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | print_le_oob(shell, &oob_local); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | #endif /* CONFIG_BT_PERIPHERAL */ | 
|  |  | 
|  | #if defined(CONFIG_BT_PER_ADV) | 
|  | static int cmd_per_adv(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_le_ext_adv *adv = adv_sets[selected_adv]; | 
|  |  | 
|  | if (!adv) { | 
|  | shell_error(shell, "No extended advertisement set selected"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | if (!strcmp(argv[1], "off")) { | 
|  | if (bt_le_per_adv_stop(adv) < 0) { | 
|  | shell_error(shell, | 
|  | "Failed to stop periodic advertising"); | 
|  | } else { | 
|  | shell_print(shell, "Periodic advertising stopped"); | 
|  | } | 
|  | } else if (!strcmp(argv[1], "on")) { | 
|  | if (bt_le_per_adv_start(adv) < 0) { | 
|  | shell_error(shell, | 
|  | "Failed to start periodic advertising"); | 
|  | } else { | 
|  | shell_print(shell, "Periodic advertising started"); | 
|  | } | 
|  | } else { | 
|  | shell_error(shell, "Invalid argument: %s", argv[1]); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_per_adv_param(const struct shell *shell, size_t argc, | 
|  | char *argv[]) | 
|  | { | 
|  | struct bt_le_ext_adv *adv = adv_sets[selected_adv]; | 
|  | struct bt_le_per_adv_param param; | 
|  | int err; | 
|  |  | 
|  | if (!adv) { | 
|  | shell_error(shell, "No extended advertisement set selected"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | if (argc > 1) { | 
|  | param.interval_min = strtol(argv[1], NULL, 16); | 
|  | } else { | 
|  | param.interval_min = BT_GAP_ADV_SLOW_INT_MIN; | 
|  | } | 
|  |  | 
|  | if (argc > 2) { | 
|  | param.interval_max = strtol(argv[2], NULL, 16); | 
|  | } else { | 
|  | param.interval_max = param.interval_min * 1.2; | 
|  |  | 
|  | } | 
|  |  | 
|  | if (param.interval_min > param.interval_max) { | 
|  | shell_error(shell, | 
|  | "Min interval shall be less than max interval"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | if (argc > 3 && !strcmp(argv[3], "tx-power")) { | 
|  | param.options = BT_LE_ADV_OPT_USE_TX_POWER; | 
|  | } else { | 
|  | param.options = 0; | 
|  | } | 
|  |  | 
|  | err = bt_le_per_adv_set_param(adv, ¶m); | 
|  | if (err) { | 
|  | shell_error(shell, "Failed to set periodic advertising " | 
|  | "parameters (%d)", err); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_per_adv_data(const struct shell *shell, size_t argc, | 
|  | char *argv[]) | 
|  | { | 
|  | struct bt_le_ext_adv *adv = adv_sets[selected_adv]; | 
|  | static struct bt_data ad; | 
|  | static uint8_t hex_data[256]; | 
|  | uint8_t ad_len = 0; | 
|  | int err; | 
|  |  | 
|  | if (!adv) { | 
|  | shell_error(shell, "No extended advertisement set selected"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | memset(hex_data, 0, sizeof(hex_data)); | 
|  | ad_len = hex2bin(argv[1], strlen(argv[1]), hex_data, sizeof(hex_data)); | 
|  |  | 
|  | if (!ad_len) { | 
|  | shell_error(shell, "Could not parse adv data"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | ad.data_len = hex_data[0]; | 
|  | ad.type = hex_data[1]; | 
|  | ad.data = &hex_data[2]; | 
|  |  | 
|  | err = bt_le_per_adv_set_data(adv, &ad, 1); | 
|  | if (err) { | 
|  | shell_error(shell, | 
|  | "Failed to set periodic advertising data (%d)", | 
|  | err); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | #endif /* CONFIG_BT_PER_ADV */ | 
|  | #endif /* CONFIG_BT_EXT_ADV */ | 
|  | #endif /* CONFIG_BT_BROADCASTER */ | 
|  |  | 
|  | #if defined(CONFIG_BT_PER_ADV_SYNC) | 
|  |  | 
|  | static int cmd_per_adv_sync_create(const struct shell *shell, size_t argc, | 
|  | char *argv[]) | 
|  | { | 
|  | int err; | 
|  | struct bt_le_per_adv_sync_param create_params = { 0 }; | 
|  | uint32_t options = 0; | 
|  | struct bt_le_per_adv_sync **free_per_adv_sync = NULL; | 
|  | int i = 0; | 
|  |  | 
|  | for (i = 0; i < ARRAY_SIZE(per_adv_syncs); i++) { | 
|  | if (per_adv_syncs[i] == NULL) { | 
|  | free_per_adv_sync = &per_adv_syncs[i]; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (i == ARRAY_SIZE(per_adv_syncs)) { | 
|  | shell_error(shell, "Cannot create more per adv syncs"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | err = bt_addr_le_from_str(argv[1], argv[2], &create_params.addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid peer address (err %d)", err); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | /* Default values */ | 
|  | create_params.timeout = 1000; /* 10 seconds */ | 
|  | create_params.skip = 10; | 
|  |  | 
|  | create_params.sid = strtol(argv[3], NULL, 16); | 
|  |  | 
|  | for (int i = 4; i < argc; i++) { | 
|  | if (!strcmp(argv[i], "aoa")) { | 
|  | options |= BT_LE_PER_ADV_SYNC_OPT_DONT_SYNC_AOA; | 
|  | } else if (!strcmp(argv[i], "aod_1us")) { | 
|  | options |= BT_LE_PER_ADV_SYNC_OPT_DONT_SYNC_AOD_1US; | 
|  | } else if (!strcmp(argv[i], "aod_2us")) { | 
|  | options |= BT_LE_PER_ADV_SYNC_OPT_DONT_SYNC_AOD_2US; | 
|  | } else if (!strcmp(argv[i], "only_cte")) { | 
|  | options |= | 
|  | BT_LE_PER_ADV_SYNC_OPT_SYNC_ONLY_CONST_TONE_EXT; | 
|  | } else if (!strcmp(argv[i], "timeout")) { | 
|  | if (++i == argc) { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | create_params.timeout = strtoul(argv[i], NULL, 16); | 
|  | } else if (!strcmp(argv[i], "skip")) { | 
|  | if (++i == argc) { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | create_params.skip = strtoul(argv[i], NULL, 16); | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | /* TODO: add support to parse using the per adv list */ | 
|  | } | 
|  |  | 
|  | create_params.options = options; | 
|  |  | 
|  | err = bt_le_per_adv_sync_create(&create_params, free_per_adv_sync); | 
|  | if (err) { | 
|  | shell_error(shell, "Per adv sync failed (%d)", err); | 
|  | } else { | 
|  | shell_print(shell, "Per adv sync pending"); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_per_adv_sync_delete(const struct shell *shell, size_t argc, | 
|  | char *argv[]) | 
|  | { | 
|  | int err; | 
|  | int index = 0; | 
|  | struct bt_le_per_adv_sync **per_adv_sync = NULL; | 
|  |  | 
|  | if (argc > 1) { | 
|  | index = strtol(argv[1], NULL, 10); | 
|  | } | 
|  |  | 
|  | per_adv_sync = &per_adv_syncs[index]; | 
|  |  | 
|  | if (!per_adv_sync) { | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | err = bt_le_per_adv_sync_delete(*per_adv_sync); | 
|  |  | 
|  | if (err) { | 
|  | shell_error(shell, "Per adv sync delete failed (%d)", err); | 
|  | } else { | 
|  | shell_print(shell, "Per adv sync deleted"); | 
|  | *per_adv_sync = NULL; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_past_subscribe(const struct shell *shell, size_t argc, | 
|  | char *argv[]) | 
|  | { | 
|  | struct bt_le_per_adv_sync_transfer_param param; | 
|  | int err; | 
|  | int i = 0; | 
|  | bool global = true; | 
|  |  | 
|  | if (i == ARRAY_SIZE(per_adv_syncs)) { | 
|  | shell_error(shell, "Cannot create more per adv syncs"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | /* Default values */ | 
|  | param.options = 0; | 
|  | param.timeout = 1000; /* 10 seconds */ | 
|  | param.skip = 10; | 
|  |  | 
|  | for (int i = 1; i < argc; i++) { | 
|  | if (!strcmp(argv[i], "aoa")) { | 
|  | param.options |= | 
|  | BT_LE_PER_ADV_SYNC_TRANSFER_OPT_SYNC_NO_AOA; | 
|  | } else if (!strcmp(argv[i], "aod_1us")) { | 
|  | param.options |= | 
|  | BT_LE_PER_ADV_SYNC_TRANSFER_OPT_SYNC_NO_AOD_1US; | 
|  | } else if (!strcmp(argv[i], "aod_2us")) { | 
|  | param.options |= | 
|  | BT_LE_PER_ADV_SYNC_TRANSFER_OPT_SYNC_NO_AOD_2US; | 
|  | } else if (!strcmp(argv[i], "only_cte")) { | 
|  | param.options |= | 
|  | BT_LE_PER_ADV_SYNC_TRANSFER_OPT_SYNC_ONLY_CTE; | 
|  | } else if (!strcmp(argv[i], "timeout")) { | 
|  | if (++i == argc) { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | param.timeout = strtoul(argv[i], NULL, 16); | 
|  | } else if (!strcmp(argv[i], "skip")) { | 
|  | if (++i == argc) { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | param.skip = strtoul(argv[i], NULL, 16); | 
|  | } else if (!strcmp(argv[i], "conn")) { | 
|  | if (!default_conn) { | 
|  | shell_print(shell, "Not connected"); | 
|  | return -EINVAL; | 
|  | } | 
|  | global = false; | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  | } | 
|  |  | 
|  | bt_le_per_adv_sync_cb_register(&per_adv_sync_cb); | 
|  |  | 
|  | err = bt_le_per_adv_sync_transfer_subscribe( | 
|  | global ? NULL : default_conn, ¶m); | 
|  |  | 
|  | if (err) { | 
|  | shell_error(shell, "PAST subscribe failed (%d)", err); | 
|  | } else { | 
|  | shell_print(shell, "Subscribed to PAST"); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_past_unsubscribe(const struct shell *shell, size_t argc, | 
|  | char *argv[]) | 
|  | { | 
|  | int err; | 
|  |  | 
|  | if (argc > 1) { | 
|  | if (!strcmp(argv[1], "conn")) { | 
|  | if (default_conn) { | 
|  | err = | 
|  | bt_le_per_adv_sync_transfer_unsubscribe( | 
|  | default_conn); | 
|  | } else { | 
|  | shell_print(shell, "Not connected"); | 
|  | return -EINVAL; | 
|  | } | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  | } else { | 
|  | err = bt_le_per_adv_sync_transfer_unsubscribe(NULL); | 
|  | } | 
|  |  | 
|  | if (err) { | 
|  | shell_error(shell, "PAST unsubscribe failed (%d)", err); | 
|  | } | 
|  |  | 
|  | return err; | 
|  | } | 
|  | #endif /* CONFIG_BT_PER_ADV_SYNC */ | 
|  |  | 
|  | #if defined(CONFIG_BT_CONN) | 
|  | #if defined(CONFIG_BT_CENTRAL) | 
|  | static int cmd_connect_le(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | int err; | 
|  | bt_addr_le_t addr; | 
|  | struct bt_conn *conn; | 
|  | uint32_t options = 0; | 
|  |  | 
|  | err = bt_addr_le_from_str(argv[1], argv[2], &addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid peer address (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_EXT_ADV) | 
|  | for (size_t argn = 3; argn < argc; argn++) { | 
|  | const char *arg = argv[argn]; | 
|  |  | 
|  | if (!strcmp(arg, "coded")) { | 
|  | options |= BT_CONN_LE_OPT_CODED; | 
|  | } else if (!strcmp(arg, "no-1m")) { | 
|  | options |= BT_CONN_LE_OPT_NO_1M; | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  | } | 
|  | #endif /* defined(CONFIG_BT_EXT_ADV) */ | 
|  |  | 
|  | struct bt_conn_le_create_param *create_params = | 
|  | BT_CONN_LE_CREATE_PARAM(options, | 
|  | BT_GAP_SCAN_FAST_INTERVAL, | 
|  | BT_GAP_SCAN_FAST_INTERVAL); | 
|  |  | 
|  | err = bt_conn_le_create(&addr, create_params, BT_LE_CONN_PARAM_DEFAULT, | 
|  | &conn); | 
|  | if (err) { | 
|  | shell_error(shell, "Connection failed (%d)", err); | 
|  | return -ENOEXEC; | 
|  | } else { | 
|  |  | 
|  | shell_print(shell, "Connection pending"); | 
|  |  | 
|  | /* unref connection obj in advance as app user */ | 
|  | bt_conn_unref(conn); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #if !defined(CONFIG_BT_WHITELIST) | 
|  | static int cmd_auto_conn(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | bt_addr_le_t addr; | 
|  | int err; | 
|  |  | 
|  | err = bt_addr_le_from_str(argv[1], argv[2], &addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid peer address (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | if (argc < 4) { | 
|  | return bt_le_set_auto_conn(&addr, BT_LE_CONN_PARAM_DEFAULT); | 
|  | } else if (!strcmp(argv[3], "on")) { | 
|  | return bt_le_set_auto_conn(&addr, BT_LE_CONN_PARAM_DEFAULT); | 
|  | } else if (!strcmp(argv[3], "off")) { | 
|  | return bt_le_set_auto_conn(&addr, NULL); | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | #endif /* !defined(CONFIG_BT_WHITELIST) */ | 
|  | #endif /* CONFIG_BT_CENTRAL */ | 
|  |  | 
|  | static int cmd_disconnect(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_conn *conn; | 
|  | int err; | 
|  |  | 
|  | if (default_conn && argc < 3) { | 
|  | conn = bt_conn_ref(default_conn); | 
|  | } else { | 
|  | bt_addr_le_t addr; | 
|  |  | 
|  | if (argc < 3) { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | err = bt_addr_le_from_str(argv[1], argv[2], &addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid peer address (err %d)", | 
|  | err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | conn = bt_conn_lookup_addr_le(selected_id, &addr); | 
|  | } | 
|  |  | 
|  | if (!conn) { | 
|  | shell_error(shell, "Not connected"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | err = bt_conn_disconnect(conn, BT_HCI_ERR_REMOTE_USER_TERM_CONN); | 
|  | if (err) { | 
|  | shell_error(shell, "Disconnection failed (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | bt_conn_unref(conn); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_select(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_conn *conn; | 
|  | bt_addr_le_t addr; | 
|  | int err; | 
|  |  | 
|  | err = bt_addr_le_from_str(argv[1], argv[2], &addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid peer address (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | conn = bt_conn_lookup_addr_le(selected_id, &addr); | 
|  | if (!conn) { | 
|  | shell_error(shell, "No matching connection found"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | if (default_conn) { | 
|  | bt_conn_unref(default_conn); | 
|  | } | 
|  |  | 
|  | default_conn = conn; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const char *get_conn_type_str(uint8_t type) | 
|  | { | 
|  | switch (type) { | 
|  | case BT_CONN_TYPE_LE: return "LE"; | 
|  | case BT_CONN_TYPE_BR: return "BR/EDR"; | 
|  | case BT_CONN_TYPE_SCO: return "SCO"; | 
|  | default: return "Invalid"; | 
|  | } | 
|  | } | 
|  |  | 
|  | static const char *get_conn_role_str(uint8_t role) | 
|  | { | 
|  | switch (role) { | 
|  | case BT_CONN_ROLE_MASTER: return "master"; | 
|  | case BT_CONN_ROLE_SLAVE: return "slave"; | 
|  | default: return "Invalid"; | 
|  | } | 
|  | } | 
|  |  | 
|  | static void print_le_addr(const char *desc, const bt_addr_le_t *addr) | 
|  | { | 
|  | char addr_str[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | const char *addr_desc = bt_addr_le_is_identity(addr) ? "identity" : | 
|  | bt_addr_le_is_rpa(addr) ? "resolvable" : | 
|  | "non-resolvable"; | 
|  |  | 
|  | bt_addr_le_to_str(addr, addr_str, sizeof(addr_str)); | 
|  |  | 
|  | shell_print(ctx_shell, "%s address: %s (%s)", desc, addr_str, | 
|  | addr_desc); | 
|  | } | 
|  |  | 
|  | static int cmd_info(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_conn *conn = NULL; | 
|  | struct bt_conn_info info; | 
|  | bt_addr_le_t addr; | 
|  | int err; | 
|  |  | 
|  | switch (argc) { | 
|  | case 1: | 
|  | if (default_conn) { | 
|  | conn = bt_conn_ref(default_conn); | 
|  | } | 
|  | break; | 
|  | case 2: | 
|  | addr.type = BT_ADDR_LE_PUBLIC; | 
|  | err = bt_addr_from_str(argv[1], &addr.a); | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid peer address (err %d)", | 
|  | err); | 
|  | return err; | 
|  | } | 
|  | conn = bt_conn_lookup_addr_le(selected_id, &addr); | 
|  | break; | 
|  | case 3: | 
|  | err = bt_addr_le_from_str(argv[1], argv[2], &addr); | 
|  |  | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid peer address (err %d)", | 
|  | err); | 
|  | return err; | 
|  | } | 
|  | conn = bt_conn_lookup_addr_le(selected_id, &addr); | 
|  | break; | 
|  | } | 
|  |  | 
|  | if (!conn) { | 
|  | shell_error(shell, "Not connected"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | err = bt_conn_get_info(conn, &info); | 
|  | if (err) { | 
|  | shell_print(ctx_shell, "Failed to get info"); | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | shell_print(ctx_shell, "Type: %s, Role: %s, Id: %u", | 
|  | get_conn_type_str(info.type), | 
|  | get_conn_role_str(info.role), | 
|  | info.id); | 
|  |  | 
|  | if (info.type == BT_CONN_TYPE_LE) { | 
|  | print_le_addr("Remote", info.le.dst); | 
|  | print_le_addr("Local", info.le.src); | 
|  | print_le_addr("Remote on-air", info.le.remote); | 
|  | print_le_addr("Local on-air", info.le.local); | 
|  |  | 
|  | shell_print(ctx_shell, "Interval: 0x%04x (%u ms)", | 
|  | info.le.interval, | 
|  | BT_INTERVAL_TO_MS(info.le.interval)); | 
|  | shell_print(ctx_shell, "Latency: 0x%04x (%u ms)", | 
|  | info.le.latency, | 
|  | BT_INTERVAL_TO_MS(info.le.latency)); | 
|  | shell_print(ctx_shell, "Supervision timeout: 0x%04x (%d ms)", | 
|  | info.le.timeout, info.le.timeout * 10); | 
|  | #if defined(CONFIG_BT_USER_PHY_UPDATE) | 
|  | shell_print(ctx_shell, "LE PHY: TX PHY %s, RX PHY %s", | 
|  | phy2str(info.le.phy->tx_phy), | 
|  | phy2str(info.le.phy->rx_phy)); | 
|  | #endif | 
|  | #if defined(CONFIG_BT_USER_DATA_LEN_UPDATE) | 
|  | shell_print(ctx_shell, "LE data len: TX (len: %d time: %d)" | 
|  | " RX (len: %d time: %d)", | 
|  | info.le.data_len->tx_max_len, | 
|  | info.le.data_len->tx_max_time, | 
|  | info.le.data_len->rx_max_len, | 
|  | info.le.data_len->rx_max_time); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_BREDR) | 
|  | if (info.type == BT_CONN_TYPE_BR) { | 
|  | char addr_str[BT_ADDR_STR_LEN]; | 
|  |  | 
|  | bt_addr_to_str(info.br.dst, addr_str, sizeof(addr_str)); | 
|  | shell_print(ctx_shell, "Peer address %s", addr_str); | 
|  | } | 
|  | #endif /* defined(CONFIG_BT_BREDR) */ | 
|  |  | 
|  | done: | 
|  | bt_conn_unref(conn); | 
|  |  | 
|  | return err; | 
|  | } | 
|  |  | 
|  | static int cmd_conn_update(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_le_conn_param param; | 
|  | int err; | 
|  |  | 
|  | param.interval_min = strtoul(argv[1], NULL, 16); | 
|  | param.interval_max = strtoul(argv[2], NULL, 16); | 
|  | param.latency = strtoul(argv[3], NULL, 16); | 
|  | param.timeout = strtoul(argv[4], NULL, 16); | 
|  |  | 
|  | err = bt_conn_le_param_update(default_conn, ¶m); | 
|  | if (err) { | 
|  | shell_error(shell, "conn update failed (err %d).", err); | 
|  | } else { | 
|  | shell_print(shell, "conn update initiated."); | 
|  | } | 
|  |  | 
|  | return err; | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_USER_DATA_LEN_UPDATE) | 
|  | static uint16_t tx_time_calc(uint8_t phy, uint16_t max_len) | 
|  | { | 
|  | /* Access address + header + payload + MIC + CRC */ | 
|  | uint16_t total_len = 4 + 2 + max_len + 4 + 3; | 
|  |  | 
|  | switch (phy) { | 
|  | case BT_GAP_LE_PHY_1M: | 
|  | /* 1 byte preamble, 8 us per byte */ | 
|  | return 8 * (1 + total_len); | 
|  | case BT_GAP_LE_PHY_2M: | 
|  | /* 2 byte preamble, 4 us per byte */ | 
|  | return 4 * (2 + total_len); | 
|  | case BT_GAP_LE_PHY_CODED: | 
|  | /* S8: Preamble + CI + TERM1 + 64 us per byte + TERM2 */ | 
|  | return 80 + 16 + 24 + 64 * (total_len) + 24; | 
|  | default: | 
|  | return 0; | 
|  | } | 
|  | } | 
|  |  | 
|  | static int cmd_conn_data_len_update(const struct shell *shell, size_t argc, | 
|  | char *argv[]) | 
|  | { | 
|  | struct bt_conn_le_data_len_param param; | 
|  | int err; | 
|  |  | 
|  | param.tx_max_len = strtoul(argv[1], NULL, 10); | 
|  |  | 
|  | if (argc > 2) { | 
|  | param.tx_max_time = strtoul(argv[2], NULL, 10); | 
|  | } else { | 
|  | /* Assume 1M if not able to retrieve PHY */ | 
|  | uint8_t phy = BT_GAP_LE_PHY_1M; | 
|  |  | 
|  | #if defined(CONFIG_BT_USER_PHY_UPDATE) | 
|  | struct bt_conn_info info; | 
|  |  | 
|  | err = bt_conn_get_info(default_conn, &info); | 
|  | if (!err) { | 
|  | phy = info.le.phy->tx_phy; | 
|  | } | 
|  | #endif | 
|  | param.tx_max_time = tx_time_calc(phy, param.tx_max_len); | 
|  | shell_print(shell, "Calculated tx time: %d", param.tx_max_time); | 
|  | } | 
|  |  | 
|  |  | 
|  |  | 
|  | err = bt_conn_le_data_len_update(default_conn, ¶m); | 
|  | if (err) { | 
|  | shell_error(shell, "data len update failed (err %d).", err); | 
|  | } else { | 
|  | shell_print(shell, "data len update initiated."); | 
|  | } | 
|  |  | 
|  | return err; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if defined(CONFIG_BT_USER_PHY_UPDATE) | 
|  | static int cmd_conn_phy_update(const struct shell *shell, size_t argc, | 
|  | char *argv[]) | 
|  | { | 
|  | struct bt_conn_le_phy_param param; | 
|  | int err; | 
|  |  | 
|  | param.pref_tx_phy = strtoul(argv[1], NULL, 16); | 
|  | param.pref_rx_phy = param.pref_tx_phy; | 
|  | param.options = BT_CONN_LE_PHY_OPT_NONE; | 
|  |  | 
|  | for (size_t argn = 2; argn < argc; argn++) { | 
|  | const char *arg = argv[argn]; | 
|  |  | 
|  | if (!strcmp(arg, "s2")) { | 
|  | param.options |= BT_CONN_LE_PHY_OPT_CODED_S2; | 
|  | } else if (!strcmp(arg, "s8")) { | 
|  | param.options |= BT_CONN_LE_PHY_OPT_CODED_S8; | 
|  | } else { | 
|  | param.pref_rx_phy = strtoul(arg, NULL, 16); | 
|  | } | 
|  | } | 
|  |  | 
|  | err = bt_conn_le_phy_update(default_conn, ¶m); | 
|  | if (err) { | 
|  | shell_error(shell, "PHY update failed (err %d).", err); | 
|  | } else { | 
|  | shell_print(shell, "PHY update initiated."); | 
|  | } | 
|  |  | 
|  | return err; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if defined(CONFIG_BT_CENTRAL) | 
|  | static int cmd_chan_map(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | uint8_t chan_map[5] = {}; | 
|  | int err; | 
|  |  | 
|  | if (hex2bin(argv[1], strlen(argv[1]), chan_map, 5) == 0) { | 
|  | shell_error(shell, "Invalid channel map"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  | sys_mem_swap(chan_map, 5); | 
|  |  | 
|  | err = bt_le_set_chan_map(chan_map); | 
|  | if (err) { | 
|  | shell_error(shell, "Failed to set channel map (err %d)", err); | 
|  | } else { | 
|  | shell_print(shell, "Channel map set"); | 
|  | } | 
|  |  | 
|  | return err; | 
|  | } | 
|  | #endif /* CONFIG_BT_CENTRAL */ | 
|  |  | 
|  | static int cmd_oob(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | int err; | 
|  |  | 
|  | err = bt_le_oob_get_local(selected_id, &oob_local); | 
|  | if (err) { | 
|  | shell_error(shell, "OOB data failed"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | print_le_oob(shell, &oob_local); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_SMP) || defined(CONFIG_BT_BREDR) | 
|  | static int cmd_oob_remote(const struct shell *shell, size_t argc, | 
|  | char *argv[]) | 
|  | { | 
|  | int err; | 
|  | bt_addr_le_t addr; | 
|  |  | 
|  | err = bt_addr_le_from_str(argv[1], argv[2], &addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid peer address (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | bt_addr_le_copy(&oob_remote.addr, &addr); | 
|  |  | 
|  | if (argc == 5) { | 
|  | hex2bin(argv[3], strlen(argv[3]), oob_remote.le_sc_data.r, | 
|  | sizeof(oob_remote.le_sc_data.r)); | 
|  | hex2bin(argv[4], strlen(argv[4]), oob_remote.le_sc_data.c, | 
|  | sizeof(oob_remote.le_sc_data.c)); | 
|  | bt_set_oob_data_flag(true); | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_oob_clear(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | memset(&oob_remote, 0, sizeof(oob_remote)); | 
|  | bt_set_oob_data_flag(false); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | #endif /* CONFIG_BT_SMP || CONFIG_BT_BREDR) */ | 
|  |  | 
|  | static int cmd_clear(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | bt_addr_le_t addr; | 
|  | int err; | 
|  |  | 
|  | if (strcmp(argv[1], "all") == 0) { | 
|  | err = bt_unpair(selected_id, NULL); | 
|  | if (err) { | 
|  | shell_error(shell, "Failed to clear pairings (err %d)", | 
|  | err); | 
|  | return err; | 
|  | } else { | 
|  | shell_print(shell, "Pairings successfully cleared"); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | if (argc < 3) { | 
|  | #if defined(CONFIG_BT_BREDR) | 
|  | addr.type = BT_ADDR_LE_PUBLIC; | 
|  | err = bt_addr_from_str(argv[1], &addr.a); | 
|  | #else | 
|  | shell_print(shell, "Both address and address type needed"); | 
|  | return -ENOEXEC; | 
|  | #endif | 
|  | } else { | 
|  | err = bt_addr_le_from_str(argv[1], argv[2], &addr); | 
|  | } | 
|  |  | 
|  | if (err) { | 
|  | shell_print(shell, "Invalid address"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | err = bt_unpair(selected_id, &addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Failed to clear pairing (err %d)", err); | 
|  | } else { | 
|  | shell_print(shell, "Pairing successfully cleared"); | 
|  | } | 
|  |  | 
|  | return err; | 
|  | } | 
|  | #endif /* CONFIG_BT_CONN */ | 
|  |  | 
|  | #if defined(CONFIG_BT_SMP) || defined(CONFIG_BT_BREDR) | 
|  | static int cmd_security(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | int err, sec; | 
|  | struct bt_conn_info info; | 
|  |  | 
|  | if (!default_conn || (bt_conn_get_info(default_conn, &info) < 0)) { | 
|  | shell_error(shell, "Not connected"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | sec = *argv[1] - '0'; | 
|  |  | 
|  | if ((info.type == BT_CONN_TYPE_BR && | 
|  | (sec < BT_SECURITY_L0 || sec > BT_SECURITY_L3))) { | 
|  | shell_error(shell, "Invalid BR/EDR security level (%d)", sec); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | if ((info.type == BT_CONN_TYPE_LE && | 
|  | (sec < BT_SECURITY_L1 || sec > BT_SECURITY_L4))) { | 
|  | shell_error(shell, "Invalid LE security level (%d)", sec); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | if (argc > 2) { | 
|  | if (!strcmp(argv[2], "force-pair")) { | 
|  | sec |= BT_SECURITY_FORCE_PAIR; | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return -ENOEXEC; | 
|  | } | 
|  | } | 
|  |  | 
|  | err = bt_conn_set_security(default_conn, sec); | 
|  | if (err) { | 
|  | shell_error(shell, "Setting security failed (err %d)", err); | 
|  | } | 
|  |  | 
|  | return err; | 
|  | } | 
|  |  | 
|  | static int cmd_bondable(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | const char *bondable; | 
|  |  | 
|  | bondable = argv[1]; | 
|  | if (!strcmp(bondable, "on")) { | 
|  | bt_set_bondable(true); | 
|  | } else if (!strcmp(bondable, "off")) { | 
|  | bt_set_bondable(false); | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void bond_info(const struct bt_bond_info *info, void *user_data) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  | int *bond_count = user_data; | 
|  |  | 
|  | bt_addr_le_to_str(&info->addr, addr, sizeof(addr)); | 
|  | shell_print(ctx_shell, "Remote Identity: %s", addr); | 
|  | (*bond_count)++; | 
|  | } | 
|  |  | 
|  | static int cmd_bonds(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | int bond_count = 0; | 
|  |  | 
|  | shell_print(shell, "Bonded devices:"); | 
|  | bt_foreach_bond(selected_id, bond_info, &bond_count); | 
|  | shell_print(shell, "Total %d", bond_count); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void connection_info(struct bt_conn *conn, void *user_data) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  | int *conn_count = user_data; | 
|  |  | 
|  | conn_addr_str(conn, addr, sizeof(addr)); | 
|  | shell_print(ctx_shell, "Remote Identity: %s", addr); | 
|  | (*conn_count)++; | 
|  | } | 
|  |  | 
|  | static int cmd_connections(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | int conn_count = 0; | 
|  |  | 
|  | shell_print(shell, "Connected devices:"); | 
|  | bt_conn_foreach(BT_CONN_TYPE_ALL, connection_info, &conn_count); | 
|  | shell_print(shell, "Total %d", conn_count); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void auth_passkey_display(struct bt_conn *conn, unsigned int passkey) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  | char passkey_str[7]; | 
|  |  | 
|  | bt_addr_le_to_str(bt_conn_get_dst(conn), addr, sizeof(addr)); | 
|  |  | 
|  | snprintk(passkey_str, 7, "%06u", passkey); | 
|  |  | 
|  | shell_print(ctx_shell, "Passkey for %s: %s", addr, passkey_str); | 
|  | } | 
|  |  | 
|  | static void auth_passkey_confirm(struct bt_conn *conn, unsigned int passkey) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  | char passkey_str[7]; | 
|  |  | 
|  | bt_addr_le_to_str(bt_conn_get_dst(conn), addr, sizeof(addr)); | 
|  |  | 
|  | snprintk(passkey_str, 7, "%06u", passkey); | 
|  |  | 
|  | shell_print(ctx_shell, "Confirm passkey for %s: %s", addr, passkey_str); | 
|  | } | 
|  |  | 
|  | static void auth_passkey_entry(struct bt_conn *conn) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(bt_conn_get_dst(conn), addr, sizeof(addr)); | 
|  |  | 
|  | shell_print(ctx_shell, "Enter passkey for %s", addr); | 
|  | } | 
|  |  | 
|  | static void auth_cancel(struct bt_conn *conn) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | conn_addr_str(conn, addr, sizeof(addr)); | 
|  |  | 
|  | shell_print(ctx_shell, "Pairing cancelled: %s", addr); | 
|  |  | 
|  | /* clear connection reference for sec mode 3 pairing */ | 
|  | if (pairing_conn) { | 
|  | bt_conn_unref(pairing_conn); | 
|  | pairing_conn = NULL; | 
|  | } | 
|  | } | 
|  |  | 
|  | static void auth_pairing_confirm(struct bt_conn *conn) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(bt_conn_get_dst(conn), addr, sizeof(addr)); | 
|  |  | 
|  | shell_print(ctx_shell, "Confirm pairing for %s", addr); | 
|  | } | 
|  |  | 
|  | #if !defined(CONFIG_BT_SMP_OOB_LEGACY_PAIR_ONLY) | 
|  | static const char *oob_config_str(int oob_config) | 
|  | { | 
|  | switch (oob_config) { | 
|  | case BT_CONN_OOB_LOCAL_ONLY: | 
|  | return "Local"; | 
|  | case BT_CONN_OOB_REMOTE_ONLY: | 
|  | return "Remote"; | 
|  | case BT_CONN_OOB_BOTH_PEERS: | 
|  | return "Local and Remote"; | 
|  | case BT_CONN_OOB_NO_DATA: | 
|  | default: | 
|  | return "no"; | 
|  | } | 
|  | } | 
|  | #endif /* !defined(CONFIG_BT_SMP_OOB_LEGACY_PAIR_ONLY) */ | 
|  |  | 
|  | static void auth_pairing_oob_data_request(struct bt_conn *conn, | 
|  | struct bt_conn_oob_info *oob_info) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  | struct bt_conn_info info; | 
|  | int err; | 
|  |  | 
|  | err = bt_conn_get_info(conn, &info); | 
|  | if (err) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | #if !defined(CONFIG_BT_SMP_OOB_LEGACY_PAIR_ONLY) | 
|  | if (oob_info->type == BT_CONN_OOB_LE_SC) { | 
|  | struct bt_le_oob_sc_data *oobd_local = | 
|  | oob_info->lesc.oob_config != BT_CONN_OOB_REMOTE_ONLY | 
|  | ? &oob_local.le_sc_data | 
|  | : NULL; | 
|  | struct bt_le_oob_sc_data *oobd_remote = | 
|  | oob_info->lesc.oob_config != BT_CONN_OOB_LOCAL_ONLY | 
|  | ? &oob_remote.le_sc_data | 
|  | : NULL; | 
|  |  | 
|  | if (oobd_remote && | 
|  | bt_addr_le_cmp(info.le.remote, &oob_remote.addr)) { | 
|  | bt_addr_le_to_str(info.le.remote, addr, sizeof(addr)); | 
|  | shell_print(ctx_shell, | 
|  | "No OOB data available for remote %s", | 
|  | addr); | 
|  | bt_conn_auth_cancel(conn); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (oobd_local && | 
|  | bt_addr_le_cmp(info.le.local, &oob_local.addr)) { | 
|  | bt_addr_le_to_str(info.le.local, addr, sizeof(addr)); | 
|  | shell_print(ctx_shell, | 
|  | "No OOB data available for local %s", | 
|  | addr); | 
|  | bt_conn_auth_cancel(conn); | 
|  | return; | 
|  | } | 
|  |  | 
|  | bt_le_oob_set_sc_data(conn, oobd_local, oobd_remote); | 
|  |  | 
|  | bt_addr_le_to_str(info.le.dst, addr, sizeof(addr)); | 
|  | shell_print(ctx_shell, "Set %s OOB SC data for %s, ", | 
|  | oob_config_str(oob_info->lesc.oob_config), addr); | 
|  | return; | 
|  | } | 
|  | #endif /* CONFIG_BT_SMP_OOB_LEGACY_PAIR_ONLY */ | 
|  |  | 
|  | bt_addr_le_to_str(info.le.dst, addr, sizeof(addr)); | 
|  | shell_print(ctx_shell, "Legacy OOB TK requested from remote %s", addr); | 
|  | } | 
|  |  | 
|  | static void auth_pairing_complete(struct bt_conn *conn, bool bonded) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(bt_conn_get_dst(conn), addr, sizeof(addr)); | 
|  |  | 
|  | shell_print(ctx_shell, "%s with %s", bonded ? "Bonded" : "Paired", | 
|  | addr); | 
|  | } | 
|  |  | 
|  | static void auth_pairing_failed(struct bt_conn *conn, enum bt_security_err err) | 
|  | { | 
|  | char addr[BT_ADDR_LE_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(bt_conn_get_dst(conn), addr, sizeof(addr)); | 
|  |  | 
|  | shell_print(ctx_shell, "Pairing failed with %s reason: %s (%d)", addr, | 
|  | security_err_str(err), err); | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_BREDR) | 
|  | static void auth_pincode_entry(struct bt_conn *conn, bool highsec) | 
|  | { | 
|  | char addr[BT_ADDR_STR_LEN]; | 
|  | struct bt_conn_info info; | 
|  |  | 
|  | if (bt_conn_get_info(conn, &info) < 0) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (info.type != BT_CONN_TYPE_BR) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | bt_addr_to_str(info.br.dst, addr, sizeof(addr)); | 
|  |  | 
|  | if (highsec) { | 
|  | shell_print(ctx_shell, "Enter 16 digits wide PIN code for %s", | 
|  | addr); | 
|  | } else { | 
|  | shell_print(ctx_shell, "Enter PIN code for %s", addr); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Save connection info since in security mode 3 (link level enforced | 
|  | * security) PIN request callback is called before connected callback | 
|  | */ | 
|  | if (!default_conn && !pairing_conn) { | 
|  | pairing_conn = bt_conn_ref(conn); | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if defined(CONFIG_BT_SMP_APP_PAIRING_ACCEPT) | 
|  | enum bt_security_err pairing_accept( | 
|  | struct bt_conn *conn, const struct bt_conn_pairing_feat *const feat) | 
|  | { | 
|  | shell_print(ctx_shell, "Remote pairing features: " | 
|  | "IO: 0x%02x, OOB: %d, AUTH: 0x%02x, Key: %d, " | 
|  | "Init Kdist: 0x%02x, Resp Kdist: 0x%02x", | 
|  | feat->io_capability, feat->oob_data_flag, | 
|  | feat->auth_req, feat->max_enc_key_size, | 
|  | feat->init_key_dist, feat->resp_key_dist); | 
|  |  | 
|  | return BT_SECURITY_ERR_SUCCESS; | 
|  | } | 
|  | #endif /* CONFIG_BT_SMP_APP_PAIRING_ACCEPT */ | 
|  |  | 
|  | void bond_deleted(uint8_t id, const bt_addr_le_t *peer) | 
|  | { | 
|  | char addr[BT_ADDR_STR_LEN]; | 
|  |  | 
|  | bt_addr_le_to_str(peer, addr, sizeof(addr)); | 
|  | shell_print(ctx_shell, "Bond deleted for %s, id %u", addr, id); | 
|  | } | 
|  |  | 
|  | static struct bt_conn_auth_cb auth_cb_display = { | 
|  | .passkey_display = auth_passkey_display, | 
|  | .passkey_entry = NULL, | 
|  | .passkey_confirm = NULL, | 
|  | #if defined(CONFIG_BT_BREDR) | 
|  | .pincode_entry = auth_pincode_entry, | 
|  | #endif | 
|  | .oob_data_request = NULL, | 
|  | .cancel = auth_cancel, | 
|  | .pairing_confirm = auth_pairing_confirm, | 
|  | .pairing_failed = auth_pairing_failed, | 
|  | .pairing_complete = auth_pairing_complete, | 
|  | #if defined(CONFIG_BT_SMP_APP_PAIRING_ACCEPT) | 
|  | .pairing_accept = pairing_accept, | 
|  | #endif | 
|  | .bond_deleted = bond_deleted, | 
|  | }; | 
|  |  | 
|  | static struct bt_conn_auth_cb auth_cb_display_yes_no = { | 
|  | .passkey_display = auth_passkey_display, | 
|  | .passkey_entry = NULL, | 
|  | .passkey_confirm = auth_passkey_confirm, | 
|  | #if defined(CONFIG_BT_BREDR) | 
|  | .pincode_entry = auth_pincode_entry, | 
|  | #endif | 
|  | .oob_data_request = NULL, | 
|  | .cancel = auth_cancel, | 
|  | .pairing_confirm = auth_pairing_confirm, | 
|  | .pairing_failed = auth_pairing_failed, | 
|  | .pairing_complete = auth_pairing_complete, | 
|  | #if defined(CONFIG_BT_SMP_APP_PAIRING_ACCEPT) | 
|  | .pairing_accept = pairing_accept, | 
|  | #endif | 
|  | .bond_deleted = bond_deleted, | 
|  | }; | 
|  |  | 
|  | static struct bt_conn_auth_cb auth_cb_input = { | 
|  | .passkey_display = NULL, | 
|  | .passkey_entry = auth_passkey_entry, | 
|  | .passkey_confirm = NULL, | 
|  | #if defined(CONFIG_BT_BREDR) | 
|  | .pincode_entry = auth_pincode_entry, | 
|  | #endif | 
|  | .oob_data_request = NULL, | 
|  | .cancel = auth_cancel, | 
|  | .pairing_confirm = auth_pairing_confirm, | 
|  | .pairing_failed = auth_pairing_failed, | 
|  | .pairing_complete = auth_pairing_complete, | 
|  | #if defined(CONFIG_BT_SMP_APP_PAIRING_ACCEPT) | 
|  | .pairing_accept = pairing_accept, | 
|  | #endif | 
|  | .bond_deleted = bond_deleted, | 
|  | }; | 
|  |  | 
|  | static struct bt_conn_auth_cb auth_cb_confirm = { | 
|  | #if defined(CONFIG_BT_BREDR) | 
|  | .pincode_entry = auth_pincode_entry, | 
|  | #endif | 
|  | .oob_data_request = NULL, | 
|  | .cancel = auth_cancel, | 
|  | .pairing_confirm = auth_pairing_confirm, | 
|  | .pairing_failed = auth_pairing_failed, | 
|  | .pairing_complete = auth_pairing_complete, | 
|  | #if defined(CONFIG_BT_SMP_APP_PAIRING_ACCEPT) | 
|  | .pairing_accept = pairing_accept, | 
|  | #endif | 
|  | .bond_deleted = bond_deleted, | 
|  | }; | 
|  |  | 
|  | static struct bt_conn_auth_cb auth_cb_all = { | 
|  | .passkey_display = auth_passkey_display, | 
|  | .passkey_entry = auth_passkey_entry, | 
|  | .passkey_confirm = auth_passkey_confirm, | 
|  | #if defined(CONFIG_BT_BREDR) | 
|  | .pincode_entry = auth_pincode_entry, | 
|  | #endif | 
|  | .oob_data_request = auth_pairing_oob_data_request, | 
|  | .cancel = auth_cancel, | 
|  | .pairing_confirm = auth_pairing_confirm, | 
|  | .pairing_failed = auth_pairing_failed, | 
|  | .pairing_complete = auth_pairing_complete, | 
|  | #if defined(CONFIG_BT_SMP_APP_PAIRING_ACCEPT) | 
|  | .pairing_accept = pairing_accept, | 
|  | #endif | 
|  | .bond_deleted = bond_deleted, | 
|  | }; | 
|  |  | 
|  | static struct bt_conn_auth_cb auth_cb_oob = { | 
|  | .passkey_display = NULL, | 
|  | .passkey_entry = NULL, | 
|  | .passkey_confirm = NULL, | 
|  | #if defined(CONFIG_BT_BREDR) | 
|  | .pincode_entry = NULL, | 
|  | #endif | 
|  | .oob_data_request = auth_pairing_oob_data_request, | 
|  | .cancel = auth_cancel, | 
|  | .pairing_confirm = NULL, | 
|  | .pairing_failed = auth_pairing_failed, | 
|  | .pairing_complete = auth_pairing_complete, | 
|  | #if defined(CONFIG_BT_SMP_APP_PAIRING_ACCEPT) | 
|  | .pairing_accept = pairing_accept, | 
|  | #endif | 
|  | .bond_deleted = bond_deleted, | 
|  | }; | 
|  |  | 
|  | static struct bt_conn_auth_cb auth_cb_status = { | 
|  | .pairing_failed = auth_pairing_failed, | 
|  | .pairing_complete = auth_pairing_complete, | 
|  | #if defined(CONFIG_BT_SMP_APP_PAIRING_ACCEPT) | 
|  | .pairing_accept = pairing_accept, | 
|  | #endif | 
|  | }; | 
|  |  | 
|  | static int cmd_auth(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | int err; | 
|  |  | 
|  | if (!strcmp(argv[1], "all")) { | 
|  | err = bt_conn_auth_cb_register(&auth_cb_all); | 
|  | } else if (!strcmp(argv[1], "input")) { | 
|  | err = bt_conn_auth_cb_register(&auth_cb_input); | 
|  | } else if (!strcmp(argv[1], "display")) { | 
|  | err = bt_conn_auth_cb_register(&auth_cb_display); | 
|  | } else if (!strcmp(argv[1], "yesno")) { | 
|  | err = bt_conn_auth_cb_register(&auth_cb_display_yes_no); | 
|  | } else if (!strcmp(argv[1], "confirm")) { | 
|  | err = bt_conn_auth_cb_register(&auth_cb_confirm); | 
|  | } else if (!strcmp(argv[1], "oob")) { | 
|  | err = bt_conn_auth_cb_register(&auth_cb_oob); | 
|  | } else if (!strcmp(argv[1], "status")) { | 
|  | err = bt_conn_auth_cb_register(&auth_cb_status); | 
|  | } else if (!strcmp(argv[1], "none")) { | 
|  | err = bt_conn_auth_cb_register(NULL); | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | return err; | 
|  | } | 
|  |  | 
|  | static int cmd_auth_cancel(const struct shell *shell, | 
|  | size_t argc, char *argv[]) | 
|  | { | 
|  | struct bt_conn *conn; | 
|  |  | 
|  | if (default_conn) { | 
|  | conn = default_conn; | 
|  | } else if (pairing_conn) { | 
|  | conn = pairing_conn; | 
|  | } else { | 
|  | conn = NULL; | 
|  | } | 
|  |  | 
|  | if (!conn) { | 
|  | shell_print(shell, "Not connected"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | bt_conn_auth_cancel(conn); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_auth_passkey_confirm(const struct shell *shell, | 
|  | size_t argc, char *argv[]) | 
|  | { | 
|  | if (!default_conn) { | 
|  | shell_print(shell, "Not connected"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | bt_conn_auth_passkey_confirm(default_conn); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_auth_pairing_confirm(const struct shell *shell, | 
|  | size_t argc, char *argv[]) | 
|  | { | 
|  | if (!default_conn) { | 
|  | shell_print(shell, "Not connected"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | bt_conn_auth_pairing_confirm(default_conn); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_WHITELIST) | 
|  | static int cmd_wl_add(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | bt_addr_le_t addr; | 
|  | int err; | 
|  |  | 
|  | err = bt_addr_le_from_str(argv[1], argv[2], &addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid peer address (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | err = bt_le_whitelist_add(&addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Add to whitelist failed (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_wl_rem(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | bt_addr_le_t addr; | 
|  | int err; | 
|  |  | 
|  | err = bt_addr_le_from_str(argv[1], argv[2], &addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Invalid peer address (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | err = bt_le_whitelist_rem(&addr); | 
|  | if (err) { | 
|  | shell_error(shell, "Remove from whitelist failed (err %d)", | 
|  | err); | 
|  | return err; | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_wl_clear(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | int err; | 
|  |  | 
|  | err = bt_le_whitelist_clear(); | 
|  | if (err) { | 
|  | shell_error(shell, "Clearing whitelist failed (err %d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #if defined(CONFIG_BT_CENTRAL) | 
|  | static int cmd_wl_connect(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | int err; | 
|  | const char *action = argv[1]; | 
|  | uint32_t options = 0; | 
|  |  | 
|  | #if defined(CONFIG_BT_EXT_ADV) | 
|  | for (size_t argn = 2; argn < argc; argn++) { | 
|  | const char *arg = argv[argn]; | 
|  |  | 
|  | if (!strcmp(arg, "coded")) { | 
|  | options |= BT_CONN_LE_OPT_CODED; | 
|  | } else if (!strcmp(arg, "no-1m")) { | 
|  | options |= BT_CONN_LE_OPT_NO_1M; | 
|  | } else { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  | } | 
|  | #endif /* defined(CONFIG_BT_EXT_ADV) */ | 
|  | struct bt_conn_le_create_param *create_params = | 
|  | BT_CONN_LE_CREATE_PARAM(options, | 
|  | BT_GAP_SCAN_FAST_INTERVAL, | 
|  | BT_GAP_SCAN_FAST_WINDOW); | 
|  |  | 
|  | if (!strcmp(action, "on")) { | 
|  | err = bt_conn_le_create_auto(create_params, | 
|  | BT_LE_CONN_PARAM_DEFAULT); | 
|  | if (err) { | 
|  | shell_error(shell, "Auto connect failed (err %d)", err); | 
|  | return err; | 
|  | } | 
|  | } else if (!strcmp(action, "off")) { | 
|  | err = bt_conn_create_auto_stop(); | 
|  | if (err) { | 
|  | shell_error(shell, "Auto connect stop failed (err %d)", | 
|  | err); | 
|  | } | 
|  | return err; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | #endif /* CONFIG_BT_CENTRAL */ | 
|  | #endif /* defined(CONFIG_BT_WHITELIST) */ | 
|  |  | 
|  | #if defined(CONFIG_BT_FIXED_PASSKEY) | 
|  | static int cmd_fixed_passkey(const struct shell *shell, | 
|  | size_t argc, char *argv[]) | 
|  | { | 
|  | unsigned int passkey; | 
|  | int err; | 
|  |  | 
|  | if (argc < 2) { | 
|  | bt_passkey_set(BT_PASSKEY_INVALID); | 
|  | shell_print(shell, "Fixed passkey cleared"); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | passkey = atoi(argv[1]); | 
|  | if (passkey > 999999) { | 
|  | shell_print(shell, "Passkey should be between 0-999999"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | err = bt_passkey_set(passkey); | 
|  | if (err) { | 
|  | shell_print(shell, "Setting fixed passkey failed (err %d)", | 
|  | err); | 
|  | } | 
|  |  | 
|  | return err; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | static int cmd_auth_passkey(const struct shell *shell, | 
|  | size_t argc, char *argv[]) | 
|  | { | 
|  | unsigned int passkey; | 
|  | int err; | 
|  |  | 
|  | if (!default_conn) { | 
|  | shell_print(shell, "Not connected"); | 
|  | return -ENOEXEC; | 
|  | } | 
|  |  | 
|  | passkey = atoi(argv[1]); | 
|  | if (passkey > 999999) { | 
|  | shell_print(shell, "Passkey should be between 0-999999"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | err = bt_conn_auth_passkey_entry(default_conn, passkey); | 
|  | if (err) { | 
|  | shell_error(shell, "Failed to set passkey (%d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #if !defined(CONFIG_BT_SMP_SC_PAIR_ONLY) | 
|  | static int cmd_auth_oob_tk(const struct shell *shell, size_t argc, char *argv[]) | 
|  | { | 
|  | uint8_t tk[16]; | 
|  | size_t len; | 
|  | int err; | 
|  |  | 
|  | len = hex2bin(argv[1], strlen(argv[1]), tk, sizeof(tk)); | 
|  | if (len != sizeof(tk)) { | 
|  | shell_error(shell, "TK should be 16 bytes"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | err = bt_le_oob_set_legacy_tk(default_conn, tk); | 
|  | if (err) { | 
|  | shell_error(shell, "Failed to set TK (%d)", err); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | #endif /* !defined(CONFIG_BT_SMP_SC_PAIR_ONLY) */ | 
|  | #endif /* CONFIG_BT_SMP) || CONFIG_BT_BREDR */ | 
|  |  | 
|  |  | 
|  | #define HELP_NONE "[none]" | 
|  | #define HELP_ADDR_LE "<address: XX:XX:XX:XX:XX:XX> <type: (public|random)>" | 
|  |  | 
|  | #if defined(CONFIG_BT_EXT_ADV) | 
|  | #define EXT_ADV_SCAN_OPT " [coded] [no-1m]" | 
|  | #define EXT_ADV_PARAM "<type: conn-scan conn-nscan, nconn-scan nconn-nscan> " \ | 
|  | "[ext-adv] [no-2m] [coded] "                            \ | 
|  | "[whitelist: wl, wl-scan, wl-conn] [identity] [name] "  \ | 
|  | "[directed "HELP_ADDR_LE"] [mode: low]"                 \ | 
|  | "[disable-37] [disable-38] [disable-39]" | 
|  | #else | 
|  | #define EXT_ADV_SCAN_OPT "" | 
|  | #endif /* defined(CONFIG_BT_EXT_ADV) */ | 
|  |  | 
|  | SHELL_STATIC_SUBCMD_SET_CREATE(bt_cmds, | 
|  | SHELL_CMD_ARG(init, NULL, "[no-settings-load], [sync]", | 
|  | cmd_init, 1, 2), | 
|  | #if defined(CONFIG_SETTINGS) | 
|  | SHELL_CMD_ARG(settings-load, NULL, HELP_NONE, cmd_settings_load, 1, 0), | 
|  | #endif | 
|  | #if defined(CONFIG_BT_HCI) | 
|  | SHELL_CMD_ARG(hci-cmd, NULL, "<ogf> <ocf> [data]", cmd_hci_cmd, 3, 1), | 
|  | #endif | 
|  | SHELL_CMD_ARG(id-create, NULL, "[addr]", cmd_id_create, 1, 1), | 
|  | SHELL_CMD_ARG(id-reset, NULL, "<id> [addr]", cmd_id_reset, 2, 1), | 
|  | SHELL_CMD_ARG(id-delete, NULL, "<id>", cmd_id_delete, 2, 0), | 
|  | SHELL_CMD_ARG(id-show, NULL, HELP_NONE, cmd_id_show, 1, 0), | 
|  | SHELL_CMD_ARG(id-select, NULL, "<id>", cmd_id_select, 2, 0), | 
|  | SHELL_CMD_ARG(name, NULL, "[name]", cmd_name, 1, 1), | 
|  | #if defined(CONFIG_BT_OBSERVER) | 
|  | SHELL_CMD_ARG(scan, NULL, | 
|  | "<value: on, passive, off> [filter: dups, nodups] [wl]" | 
|  | EXT_ADV_SCAN_OPT, | 
|  | cmd_scan, 2, 4), | 
|  | #endif /* CONFIG_BT_OBSERVER */ | 
|  | #if defined(CONFIG_BT_BROADCASTER) | 
|  | SHELL_CMD_ARG(advertise, NULL, | 
|  | "<type: off, on, scan, nconn> [mode: discov, non_discov] " | 
|  | "[whitelist: wl, wl-scan, wl-conn] [identity] [no-name] " | 
|  | "[one-time] [disable-37] [disable-38] [disable-39]", | 
|  | cmd_advertise, 2, 8), | 
|  | #if defined(CONFIG_BT_PERIPHERAL) | 
|  | SHELL_CMD_ARG(directed-adv, NULL, HELP_ADDR_LE " [mode: low] " | 
|  | "[identity] [dir-rpa]", | 
|  | cmd_directed_adv, 3, 6), | 
|  | #endif /* CONFIG_BT_PERIPHERAL */ | 
|  | #if defined(CONFIG_BT_EXT_ADV) | 
|  | SHELL_CMD_ARG(adv-create, NULL, EXT_ADV_PARAM, cmd_adv_create, 2, 11), | 
|  | SHELL_CMD_ARG(adv-param, NULL, EXT_ADV_PARAM, cmd_adv_param, 2, 11), | 
|  | SHELL_CMD_ARG(adv-data, NULL, "<data> [scan-response <data>] " | 
|  | "<type: discov, name, hex>", cmd_adv_data, | 
|  | 1, 16), | 
|  | SHELL_CMD_ARG(adv-start, NULL, | 
|  | "[timeout <timeout>] [num-events <num events>]", | 
|  | cmd_adv_start, 1, 4), | 
|  | SHELL_CMD_ARG(adv-stop, NULL, "", cmd_adv_stop, 1, 0), | 
|  | SHELL_CMD_ARG(adv-delete, NULL, "", cmd_adv_delete, 1, 0), | 
|  | SHELL_CMD_ARG(adv-select, NULL, "[adv]", cmd_adv_select, 1, 1), | 
|  | SHELL_CMD_ARG(adv-info, NULL, HELP_NONE, cmd_adv_info, 1, 0), | 
|  | #if defined(CONFIG_BT_PERIPHERAL) | 
|  | SHELL_CMD_ARG(adv-oob, NULL, HELP_NONE, cmd_adv_oob, 1, 0), | 
|  | #endif /* CONFIG_BT_PERIPHERAL */ | 
|  | #if defined(CONFIG_BT_PER_ADV) | 
|  | SHELL_CMD_ARG(per-adv, NULL, "<type: off, on>", cmd_per_adv, 2, 0), | 
|  | SHELL_CMD_ARG(per-adv-param, NULL, | 
|  | "[<interval-min> [<interval-max> [tx_power]]]", | 
|  | cmd_per_adv_param, 1, 3), | 
|  | SHELL_CMD_ARG(per-adv-data, NULL, "<data>", cmd_per_adv_data, 2, 0), | 
|  | #endif /* CONFIG_BT_PER_ADV */ | 
|  | #endif /* CONFIG_BT_EXT_ADV */ | 
|  | #endif /* CONFIG_BT_BROADCASTER */ | 
|  | #if defined(CONFIG_BT_PER_ADV_SYNC) | 
|  | SHELL_CMD_ARG(per-adv-sync-create, NULL, | 
|  | HELP_ADDR_LE " <sid> [skip <count>] [timeout <ms>] [aoa] " | 
|  | "[aod_1us] [aod_2us] [cte_only]", | 
|  | cmd_per_adv_sync_create, 4, 6), | 
|  | SHELL_CMD_ARG(per-adv-sync-delete, NULL, "[<index>]", | 
|  | cmd_per_adv_sync_delete, 1, 1), | 
|  | #endif /* defined(CONFIG_BT_PER_ADV_SYNC) */ | 
|  | #if defined(CONFIG_BT_CONN) | 
|  | #if defined(CONFIG_BT_PER_ADV_SYNC) | 
|  | SHELL_CMD_ARG(past-subscribe, NULL, "[conn] [skip <count>] " | 
|  | "[timeout <ms>] [aoa] [aod_1us] [aod_2us] [cte_only]", | 
|  | cmd_past_subscribe, 1, 7), | 
|  | SHELL_CMD_ARG(past-unsubscribe, NULL, "[conn]", | 
|  | cmd_past_unsubscribe, 1, 1), | 
|  | #endif /* defined(CONFIG_BT_PER_ADV_SYNC) */ | 
|  | #if defined(CONFIG_BT_CENTRAL) | 
|  | SHELL_CMD_ARG(connect, NULL, HELP_ADDR_LE EXT_ADV_SCAN_OPT, | 
|  | cmd_connect_le, 3, 3), | 
|  | #if !defined(CONFIG_BT_WHITELIST) | 
|  | SHELL_CMD_ARG(auto-conn, NULL, HELP_ADDR_LE, cmd_auto_conn, 3, 0), | 
|  | #endif /* !defined(CONFIG_BT_WHITELIST) */ | 
|  | #endif /* CONFIG_BT_CENTRAL */ | 
|  | SHELL_CMD_ARG(disconnect, NULL, HELP_NONE, cmd_disconnect, 1, 2), | 
|  | SHELL_CMD_ARG(select, NULL, HELP_ADDR_LE, cmd_select, 3, 0), | 
|  | SHELL_CMD_ARG(info, NULL, HELP_ADDR_LE, cmd_info, 1, 2), | 
|  | SHELL_CMD_ARG(conn-update, NULL, "<min> <max> <latency> <timeout>", | 
|  | cmd_conn_update, 5, 0), | 
|  | #if defined(CONFIG_BT_USER_DATA_LEN_UPDATE) | 
|  | SHELL_CMD_ARG(data-len-update, NULL, "<tx_max_len> [tx_max_time]", | 
|  | cmd_conn_data_len_update, 2, 1), | 
|  | #endif | 
|  | #if defined(CONFIG_BT_USER_PHY_UPDATE) | 
|  | SHELL_CMD_ARG(phy-update, NULL, "<tx_phy> [rx_phy] [s2] [s8]", | 
|  | cmd_conn_phy_update, 2, 3), | 
|  | #endif | 
|  | #if defined(CONFIG_BT_CENTRAL) | 
|  | SHELL_CMD_ARG(channel-map, NULL, "<channel-map: XXXXXXXXXX> (36-0)", | 
|  | cmd_chan_map, 2, 1), | 
|  | #endif /* CONFIG_BT_CENTRAL */ | 
|  | SHELL_CMD_ARG(oob, NULL, NULL, cmd_oob, 1, 0), | 
|  | SHELL_CMD_ARG(clear, NULL, "<remote: addr, all>", cmd_clear, 2, 1), | 
|  | #if defined(CONFIG_BT_SMP) || defined(CONFIG_BT_BREDR) | 
|  | SHELL_CMD_ARG(security, NULL, "<security level BR/EDR: 0 - 3, " | 
|  | "LE: 1 - 4> [force-pair]", | 
|  | cmd_security, 2, 1), | 
|  | SHELL_CMD_ARG(bondable, NULL, "<bondable: on, off>", cmd_bondable, | 
|  | 2, 0), | 
|  | SHELL_CMD_ARG(bonds, NULL, HELP_NONE, cmd_bonds, 1, 0), | 
|  | SHELL_CMD_ARG(connections, NULL, HELP_NONE, cmd_connections, 1, 0), | 
|  | SHELL_CMD_ARG(auth, NULL, | 
|  | "<method: all, input, display, yesno, confirm, " | 
|  | "oob, status, none>", | 
|  | cmd_auth, 2, 0), | 
|  | SHELL_CMD_ARG(auth-cancel, NULL, HELP_NONE, cmd_auth_cancel, 1, 0), | 
|  | SHELL_CMD_ARG(auth-passkey, NULL, "<passkey>", cmd_auth_passkey, 2, 0), | 
|  | SHELL_CMD_ARG(auth-passkey-confirm, NULL, HELP_NONE, | 
|  | cmd_auth_passkey_confirm, 1, 0), | 
|  | SHELL_CMD_ARG(auth-pairing-confirm, NULL, HELP_NONE, | 
|  | cmd_auth_pairing_confirm, 1, 0), | 
|  | #if !defined(CONFIG_BT_SMP_SC_PAIR_ONLY) | 
|  | SHELL_CMD_ARG(auth-oob-tk, NULL, "<tk>", cmd_auth_oob_tk, 2, 0), | 
|  | #endif /* !defined(CONFIG_BT_SMP_SC_PAIR_ONLY) */ | 
|  | SHELL_CMD_ARG(oob-remote, NULL, | 
|  | HELP_ADDR_LE" <oob rand> <oob confirm>", | 
|  | cmd_oob_remote, 3, 2), | 
|  | SHELL_CMD_ARG(oob-clear, NULL, HELP_NONE, cmd_oob_clear, 1, 0), | 
|  | #if defined(CONFIG_BT_WHITELIST) | 
|  | SHELL_CMD_ARG(wl-add, NULL, HELP_ADDR_LE, cmd_wl_add, 3, 0), | 
|  | SHELL_CMD_ARG(wl-rem, NULL, HELP_ADDR_LE, cmd_wl_rem, 3, 0), | 
|  | SHELL_CMD_ARG(wl-clear, NULL, HELP_NONE, cmd_wl_clear, 1, 0), | 
|  |  | 
|  | #if defined(CONFIG_BT_CENTRAL) | 
|  | SHELL_CMD_ARG(wl-connect, NULL, "<on, off>" EXT_ADV_SCAN_OPT, | 
|  | cmd_wl_connect, 2, 3), | 
|  | #endif /* CONFIG_BT_CENTRAL */ | 
|  | #endif /* defined(CONFIG_BT_WHITELIST) */ | 
|  | #if defined(CONFIG_BT_FIXED_PASSKEY) | 
|  | SHELL_CMD_ARG(fixed-passkey, NULL, "[passkey]", cmd_fixed_passkey, | 
|  | 1, 1), | 
|  | #endif | 
|  | #endif /* CONFIG_BT_SMP || CONFIG_BT_BREDR) */ | 
|  | #endif /* CONFIG_BT_CONN */ | 
|  | #if defined(CONFIG_BT_HCI_MESH_EXT) | 
|  | SHELL_CMD(mesh_adv, NULL, "<on, off>", cmd_mesh_adv), | 
|  | #endif /* CONFIG_BT_HCI_MESH_EXT */ | 
|  |  | 
|  | #if defined(CONFIG_BT_LL_SW_SPLIT) | 
|  | #if defined(CONFIG_BT_CTLR_ADV_EXT) | 
|  | #if defined(CONFIG_BT_BROADCASTER) | 
|  | SHELL_CMD_ARG(advx, NULL, | 
|  | "<on hdcd ldcd off> [coded] [anon] [txp] [ad]", | 
|  | cmd_advx, 2, 4), | 
|  | #endif /* CONFIG_BT_BROADCASTER */ | 
|  | #if defined(CONFIG_BT_OBSERVER) | 
|  | SHELL_CMD_ARG(scanx, NULL, "<on passive off> [coded]", cmd_scanx, | 
|  | 2, 1), | 
|  | #endif /* CONFIG_BT_OBSERVER */ | 
|  | #endif /* CONFIG_BT_CTLR_ADV_EXT */ | 
|  | #if defined(CONFIG_BT_CTLR_DTM) | 
|  | SHELL_CMD_ARG(test_tx, NULL, "<chan> <len> <type> <phy>", cmd_test_tx, | 
|  | 5, 0), | 
|  | SHELL_CMD_ARG(test_rx, NULL, "<chan> <phy> <mod_idx>", cmd_test_rx, | 
|  | 4, 0), | 
|  | SHELL_CMD_ARG(test_end, NULL, HELP_NONE, cmd_test_end, 1, 0), | 
|  | #endif /* CONFIG_BT_CTLR_DTM */ | 
|  | #endif /* CONFIG_BT_LL_SW_SPLIT */ | 
|  |  | 
|  | SHELL_SUBCMD_SET_END | 
|  | ); | 
|  |  | 
|  | static int cmd_bt(const struct shell *shell, size_t argc, char **argv) | 
|  | { | 
|  | if (argc == 1) { | 
|  | shell_help(shell); | 
|  | return SHELL_CMD_HELP_PRINTED; | 
|  | } | 
|  |  | 
|  | shell_error(shell, "%s unknown parameter: %s", argv[0], argv[1]); | 
|  |  | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | SHELL_CMD_REGISTER(bt, &bt_cmds, "Bluetooth shell commands", cmd_bt); |