483 lines
14 KiB
C
483 lines
14 KiB
C
/* hci_core.h - Bluetooth HCI core access */
|
|
|
|
/*
|
|
* Copyright (c) 2021 Nordic Semiconductor ASA
|
|
* Copyright (c) 2015-2016 Intel Corporation
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
/* LL connection parameters */
|
|
#define LE_CONN_LATENCY 0x0000
|
|
#define LE_CONN_TIMEOUT 0x002a
|
|
|
|
#if defined(CONFIG_BT_BREDR)
|
|
#define LMP_FEAT_PAGES_COUNT 3
|
|
#else
|
|
#define LMP_FEAT_PAGES_COUNT 1
|
|
#endif
|
|
|
|
/* SCO settings */
|
|
#define BT_VOICE_CVSD_16BIT 0x0060
|
|
|
|
/* k_poll event tags */
|
|
enum {
|
|
BT_EVENT_CMD_TX,
|
|
BT_EVENT_CONN_TX_QUEUE,
|
|
};
|
|
|
|
/* bt_dev flags: the flags defined here represent BT controller state */
|
|
enum {
|
|
BT_DEV_ENABLE,
|
|
BT_DEV_DISABLE,
|
|
BT_DEV_READY,
|
|
BT_DEV_PRESET_ID,
|
|
BT_DEV_HAS_PUB_KEY,
|
|
BT_DEV_PUB_KEY_BUSY,
|
|
|
|
BT_DEV_SCANNING,
|
|
BT_DEV_EXPLICIT_SCAN,
|
|
BT_DEV_ACTIVE_SCAN,
|
|
BT_DEV_SCAN_FILTER_DUP,
|
|
BT_DEV_SCAN_FILTERED,
|
|
BT_DEV_SCAN_LIMITED,
|
|
BT_DEV_INITIATING,
|
|
|
|
BT_DEV_RPA_VALID,
|
|
|
|
BT_DEV_ID_PENDING,
|
|
BT_DEV_STORE_ID,
|
|
|
|
#if defined(CONFIG_BT_BREDR)
|
|
BT_DEV_ISCAN,
|
|
BT_DEV_PSCAN,
|
|
BT_DEV_INQUIRY,
|
|
#endif /* CONFIG_BT_BREDR */
|
|
|
|
/* Total number of flags - must be at the end of the enum */
|
|
BT_DEV_NUM_FLAGS,
|
|
};
|
|
|
|
/* Flags which should not be cleared upon HCI_Reset */
|
|
#define BT_DEV_PERSISTENT_FLAGS (BIT(BT_DEV_ENABLE) | \
|
|
BIT(BT_DEV_PRESET_ID))
|
|
|
|
#if defined(CONFIG_BT_EXT_ADV_LEGACY_SUPPORT)
|
|
/* Check the feature bit for extended or legacy advertising commands */
|
|
#define BT_DEV_FEAT_LE_EXT_ADV(feat) BT_FEAT_LE_EXT_ADV(feat)
|
|
#else
|
|
/* Always use extended advertising commands. */
|
|
#define BT_DEV_FEAT_LE_EXT_ADV(feat) 1
|
|
#endif
|
|
|
|
enum {
|
|
/* Advertising set has been created in the host. */
|
|
BT_ADV_CREATED,
|
|
/* Advertising parameters has been set in the controller.
|
|
* This implies that the advertising set has been created in the
|
|
* controller.
|
|
*/
|
|
BT_ADV_PARAMS_SET,
|
|
/* Advertising data has been set in the controller. */
|
|
BT_ADV_DATA_SET,
|
|
/* Advertising random address pending to be set in the controller. */
|
|
BT_ADV_RANDOM_ADDR_PENDING,
|
|
/* The private random address of the advertiser is valid for this cycle
|
|
* of the RPA timeout.
|
|
*/
|
|
BT_ADV_RPA_VALID,
|
|
/* The advertiser set is limited by a timeout, or number of advertising
|
|
* events, or both.
|
|
*/
|
|
BT_ADV_LIMITED,
|
|
/* Advertiser set is currently advertising in the controller. */
|
|
BT_ADV_ENABLED,
|
|
/* Advertiser should include name in advertising data */
|
|
BT_ADV_INCLUDE_NAME_AD,
|
|
/* Advertiser should include name in scan response data */
|
|
BT_ADV_INCLUDE_NAME_SD,
|
|
/* Advertiser set is connectable */
|
|
BT_ADV_CONNECTABLE,
|
|
/* Advertiser set is scannable */
|
|
BT_ADV_SCANNABLE,
|
|
/* Advertiser set is using extended advertising */
|
|
BT_ADV_EXT_ADV,
|
|
/* Advertiser set has disabled the use of private addresses and is using
|
|
* the identity address instead.
|
|
*/
|
|
BT_ADV_USE_IDENTITY,
|
|
/* Advertiser has been configured to keep advertising after a connection
|
|
* has been established as long as there are connections available.
|
|
*/
|
|
BT_ADV_PERSIST,
|
|
/* Advertiser has been temporarily disabled. */
|
|
BT_ADV_PAUSED,
|
|
/* Periodic Advertising has been enabled in the controller. */
|
|
BT_PER_ADV_ENABLED,
|
|
/* Periodic Advertising parameters has been set in the controller. */
|
|
BT_PER_ADV_PARAMS_SET,
|
|
/* Constant Tone Extension parameters for Periodic Advertising
|
|
* has been set in the controller.
|
|
*/
|
|
BT_PER_ADV_CTE_PARAMS_SET,
|
|
/* Constant Tone Extension for Periodic Advertising has been enabled
|
|
* in the controller.
|
|
*/
|
|
BT_PER_ADV_CTE_ENABLED,
|
|
|
|
BT_ADV_NUM_FLAGS,
|
|
};
|
|
|
|
struct bt_le_ext_adv {
|
|
/* ID Address used for advertising */
|
|
uint8_t id;
|
|
|
|
/* Advertising handle */
|
|
uint8_t handle;
|
|
|
|
/* Current local Random Address */
|
|
bt_addr_le_t random_addr;
|
|
|
|
/* Current target address */
|
|
bt_addr_le_t target_addr;
|
|
|
|
ATOMIC_DEFINE(flags, BT_ADV_NUM_FLAGS);
|
|
|
|
#if defined(CONFIG_BT_EXT_ADV)
|
|
const struct bt_le_ext_adv_cb *cb;
|
|
|
|
/* TX Power in use by the controller */
|
|
int8_t tx_power;
|
|
#endif /* defined(CONFIG_BT_EXT_ADV) */
|
|
|
|
struct k_work_delayable lim_adv_timeout_work;
|
|
};
|
|
|
|
enum {
|
|
/** Periodic Advertising Sync has been created in the host. */
|
|
BT_PER_ADV_SYNC_CREATED,
|
|
|
|
/** Periodic Advertising Sync is established and can be terminated */
|
|
BT_PER_ADV_SYNC_SYNCED,
|
|
|
|
/** Periodic Advertising Sync is attempting to create sync */
|
|
BT_PER_ADV_SYNC_SYNCING,
|
|
|
|
/** Periodic Advertising Sync is attempting to create sync using
|
|
* Advertiser List
|
|
*/
|
|
BT_PER_ADV_SYNC_SYNCING_USE_LIST,
|
|
|
|
/** Periodic Advertising Sync established with reporting disabled */
|
|
BT_PER_ADV_SYNC_RECV_DISABLED,
|
|
|
|
/** Constant Tone Extension for Periodic Advertising has been enabled
|
|
* in the Controller.
|
|
*/
|
|
BT_PER_ADV_SYNC_CTE_ENABLED,
|
|
|
|
BT_PER_ADV_SYNC_NUM_FLAGS,
|
|
};
|
|
|
|
struct bt_le_per_adv_sync {
|
|
/** Periodic Advertiser Address */
|
|
bt_addr_le_t addr;
|
|
|
|
/** Advertiser SID */
|
|
uint8_t sid;
|
|
|
|
/** Sync handle */
|
|
uint16_t handle;
|
|
|
|
/** Periodic advertising interval (N * 1.25 ms) */
|
|
uint16_t interval;
|
|
|
|
/** Periodic advertising advertiser clock accuracy (ppm) */
|
|
uint16_t clock_accuracy;
|
|
|
|
/** Advertiser PHY */
|
|
uint8_t phy;
|
|
|
|
#if defined(CONFIG_BT_DF_CONNECTIONLESS_CTE_RX)
|
|
/**
|
|
* @brief Bitfield with allowed CTE types.
|
|
*
|
|
* Allowed values are defined by @ref bt_df_cte_type, except BT_DF_CTE_TYPE_NONE.
|
|
*/
|
|
uint8_t cte_types;
|
|
#endif /* CONFIG_BT_DF_CONNECTIONLESS_CTE_RX */
|
|
|
|
#if CONFIG_BT_PER_ADV_SYNC_BUF_SIZE > 0
|
|
/** Reassembly buffer for advertising reports */
|
|
struct net_buf_simple reassembly;
|
|
|
|
/** Storage for the reassembly buffer */
|
|
uint8_t reassembly_data[CONFIG_BT_PER_ADV_SYNC_BUF_SIZE];
|
|
#endif /* CONFIG_BT_PER_ADV_SYNC_BUF_SIZE > 0 */
|
|
|
|
/** True if the following periodic adv reports up to and
|
|
* including the next complete one should be dropped
|
|
*/
|
|
bool report_truncated;
|
|
|
|
/** Flags */
|
|
ATOMIC_DEFINE(flags, BT_PER_ADV_SYNC_NUM_FLAGS);
|
|
};
|
|
|
|
struct bt_dev_le {
|
|
/* LE features */
|
|
uint8_t features[8];
|
|
/* LE states */
|
|
uint64_t states;
|
|
|
|
#if defined(CONFIG_BT_CONN)
|
|
/* Controller buffer information */
|
|
uint16_t mtu;
|
|
struct k_sem pkts;
|
|
uint16_t acl_mtu;
|
|
struct k_sem acl_pkts;
|
|
#endif /* CONFIG_BT_CONN */
|
|
#if defined(CONFIG_BT_ISO)
|
|
uint16_t iso_mtu;
|
|
struct k_sem iso_pkts;
|
|
#endif /* CONFIG_BT_ISO */
|
|
|
|
#if defined(CONFIG_BT_SMP)
|
|
/* Size of the the controller resolving list */
|
|
uint8_t rl_size;
|
|
/* Number of entries in the resolving list. rl_entries > rl_size
|
|
* means that host-side resolving is used.
|
|
*/
|
|
uint8_t rl_entries;
|
|
#endif /* CONFIG_BT_SMP */
|
|
};
|
|
|
|
#if defined(CONFIG_BT_BREDR)
|
|
struct bt_dev_br {
|
|
/* Max controller's acceptable ACL packet length */
|
|
uint16_t mtu;
|
|
struct k_sem pkts;
|
|
uint16_t esco_pkt_type;
|
|
};
|
|
#endif
|
|
|
|
/* The theoretical max for these is 8 and 64, but there's no point
|
|
* in allocating the full memory if we only support a small subset.
|
|
* These values must be updated whenever the host implementation is
|
|
* extended beyond the current values.
|
|
*/
|
|
#define BT_DEV_VS_FEAT_MAX 1
|
|
#define BT_DEV_VS_CMDS_MAX 2
|
|
|
|
/* State tracking for the local Bluetooth controller */
|
|
struct bt_dev {
|
|
/* Local Identity Address(es) */
|
|
bt_addr_le_t id_addr[CONFIG_BT_ID_MAX];
|
|
uint8_t id_count;
|
|
|
|
struct bt_conn_le_create_param create_param;
|
|
|
|
#if !defined(CONFIG_BT_EXT_ADV)
|
|
/* Legacy advertiser */
|
|
struct bt_le_ext_adv adv;
|
|
#else
|
|
/* Pointer to reserved advertising set */
|
|
struct bt_le_ext_adv *adv;
|
|
#if (CONFIG_BT_ID_MAX > 1) && (CONFIG_BT_EXT_ADV_MAX_ADV_SET > 1)
|
|
/* When supporting multiple concurrent connectable advertising sets
|
|
* with multiple identities, we need to know the identity of
|
|
* the terminating advertising set to identify the connection object.
|
|
* The identity of the advertising set is determined by its
|
|
* advertising handle, which is part of the
|
|
* LE Set Advertising Set Terminated event which is always sent
|
|
* _after_ the LE Enhanced Connection complete event.
|
|
* Therefore we need cache this event until its identity is known.
|
|
*/
|
|
struct {
|
|
bool valid;
|
|
struct bt_hci_evt_le_enh_conn_complete evt;
|
|
} cached_conn_complete[MIN(CONFIG_BT_MAX_CONN,
|
|
CONFIG_BT_EXT_ADV_MAX_ADV_SET)];
|
|
#endif
|
|
#endif
|
|
/* Current local Random Address */
|
|
bt_addr_le_t random_addr;
|
|
uint8_t adv_conn_id;
|
|
|
|
/* Controller version & manufacturer information */
|
|
uint8_t hci_version;
|
|
uint8_t lmp_version;
|
|
uint16_t hci_revision;
|
|
uint16_t lmp_subversion;
|
|
uint16_t manufacturer;
|
|
|
|
/* LMP features (pages 0, 1, 2) */
|
|
uint8_t features[LMP_FEAT_PAGES_COUNT][8];
|
|
|
|
/* Supported commands */
|
|
uint8_t supported_commands[64];
|
|
|
|
#if defined(CONFIG_BT_HCI_VS_EXT)
|
|
/* Vendor HCI support */
|
|
uint8_t vs_features[BT_DEV_VS_FEAT_MAX];
|
|
uint8_t vs_commands[BT_DEV_VS_CMDS_MAX];
|
|
#endif
|
|
|
|
struct k_work init;
|
|
|
|
ATOMIC_DEFINE(flags, BT_DEV_NUM_FLAGS);
|
|
|
|
/* LE controller specific features */
|
|
struct bt_dev_le le;
|
|
|
|
#if defined(CONFIG_BT_BREDR)
|
|
/* BR/EDR controller specific features */
|
|
struct bt_dev_br br;
|
|
#endif
|
|
|
|
/* Number of commands controller can accept */
|
|
struct k_sem ncmd_sem;
|
|
|
|
/* Last sent HCI command */
|
|
struct net_buf *sent_cmd;
|
|
|
|
#if !defined(CONFIG_BT_RECV_IS_RX_THREAD)
|
|
/* Queue for incoming HCI events & ACL data */
|
|
struct k_fifo rx_queue;
|
|
#endif
|
|
|
|
/* Queue for outgoing HCI commands */
|
|
struct k_fifo cmd_tx_queue;
|
|
|
|
/* Registered HCI driver */
|
|
const struct bt_hci_driver *drv;
|
|
|
|
#if defined(CONFIG_BT_PRIVACY)
|
|
/* Local Identity Resolving Key */
|
|
uint8_t irk[CONFIG_BT_ID_MAX][16];
|
|
|
|
/* Work used for RPA rotation */
|
|
struct k_work_delayable rpa_update;
|
|
#endif
|
|
|
|
/* Local Name */
|
|
#if defined(CONFIG_BT_DEVICE_NAME_DYNAMIC)
|
|
char name[CONFIG_BT_DEVICE_NAME_MAX + 1];
|
|
#endif
|
|
#if defined(CONFIG_BT_DEVICE_APPEARANCE_DYNAMIC)
|
|
/* Appearance Value */
|
|
uint16_t appearance;
|
|
#endif
|
|
};
|
|
|
|
extern struct bt_dev bt_dev;
|
|
#if defined(CONFIG_BT_SMP) || defined(CONFIG_BT_BREDR)
|
|
extern const struct bt_conn_auth_cb *bt_auth;
|
|
|
|
enum bt_security_err bt_security_err_get(uint8_t hci_err);
|
|
#endif /* CONFIG_BT_SMP || CONFIG_BT_BREDR */
|
|
|
|
/* Data type to store state related with command to be updated
|
|
* when command completes successfully.
|
|
*/
|
|
struct bt_hci_cmd_state_set {
|
|
/* Target memory to be updated */
|
|
atomic_t *target;
|
|
/* Bit number to be updated in target memory */
|
|
int bit;
|
|
/* Value to determine if enable or disable bit */
|
|
bool val;
|
|
};
|
|
|
|
/* Set command state related with the command buffer */
|
|
void bt_hci_cmd_state_set_init(struct net_buf *buf,
|
|
struct bt_hci_cmd_state_set *state,
|
|
atomic_t *target, int bit, bool val);
|
|
|
|
int bt_hci_disconnect(uint16_t handle, uint8_t reason);
|
|
|
|
bool bt_le_conn_params_valid(const struct bt_le_conn_param *param);
|
|
int bt_le_set_data_len(struct bt_conn *conn, uint16_t tx_octets, uint16_t tx_time);
|
|
int bt_le_set_phy(struct bt_conn *conn, uint8_t all_phys,
|
|
uint8_t pref_tx_phy, uint8_t pref_rx_phy, uint8_t phy_opts);
|
|
uint8_t bt_get_phy(uint8_t hci_phy);
|
|
|
|
int bt_le_scan_update(bool fast_scan);
|
|
|
|
int bt_le_create_conn(const struct bt_conn *conn);
|
|
int bt_le_create_conn_cancel(void);
|
|
|
|
bool bt_addr_le_is_bonded(uint8_t id, const bt_addr_le_t *addr);
|
|
const bt_addr_le_t *bt_lookup_id_addr(uint8_t id, const bt_addr_le_t *addr);
|
|
|
|
int bt_send(struct net_buf *buf);
|
|
|
|
/* Don't require everyone to include keys.h */
|
|
struct bt_keys;
|
|
void bt_id_add(struct bt_keys *keys);
|
|
void bt_id_del(struct bt_keys *keys);
|
|
|
|
int bt_setup_random_id_addr(void);
|
|
int bt_setup_public_id_addr(void);
|
|
|
|
void bt_finalize_init(void);
|
|
|
|
void bt_hci_host_num_completed_packets(struct net_buf *buf);
|
|
|
|
/* HCI event handlers */
|
|
void bt_hci_pin_code_req(struct net_buf *buf);
|
|
void bt_hci_link_key_notify(struct net_buf *buf);
|
|
void bt_hci_link_key_req(struct net_buf *buf);
|
|
void bt_hci_io_capa_resp(struct net_buf *buf);
|
|
void bt_hci_io_capa_req(struct net_buf *buf);
|
|
void bt_hci_ssp_complete(struct net_buf *buf);
|
|
void bt_hci_user_confirm_req(struct net_buf *buf);
|
|
void bt_hci_user_passkey_notify(struct net_buf *buf);
|
|
void bt_hci_user_passkey_req(struct net_buf *buf);
|
|
void bt_hci_auth_complete(struct net_buf *buf);
|
|
|
|
/* ECC HCI event handlers */
|
|
void bt_hci_evt_le_pkey_complete(struct net_buf *buf);
|
|
void bt_hci_evt_le_dhkey_complete(struct net_buf *buf);
|
|
|
|
/* Common HCI event handlers */
|
|
void bt_hci_le_enh_conn_complete(struct bt_hci_evt_le_enh_conn_complete *evt);
|
|
|
|
/* Scan HCI event handlers */
|
|
void bt_hci_le_adv_report(struct net_buf *buf);
|
|
void bt_hci_le_scan_timeout(struct net_buf *buf);
|
|
void bt_hci_le_adv_ext_report(struct net_buf *buf);
|
|
void bt_hci_le_per_adv_sync_established(struct net_buf *buf);
|
|
void bt_hci_le_per_adv_report(struct net_buf *buf);
|
|
void bt_hci_le_per_adv_sync_lost(struct net_buf *buf);
|
|
void bt_hci_le_biginfo_adv_report(struct net_buf *buf);
|
|
void bt_hci_le_df_connectionless_iq_report(struct net_buf *buf);
|
|
void bt_hci_le_past_received(struct net_buf *buf);
|
|
|
|
/* Adv HCI event handlers */
|
|
void bt_hci_le_adv_set_terminated(struct net_buf *buf);
|
|
void bt_hci_le_scan_req_received(struct net_buf *buf);
|
|
|
|
/* BR/EDR HCI event handlers */
|
|
void bt_hci_conn_req(struct net_buf *buf);
|
|
void bt_hci_conn_complete(struct net_buf *buf);
|
|
|
|
|
|
void bt_hci_inquiry_complete(struct net_buf *buf);
|
|
void bt_hci_inquiry_result_with_rssi(struct net_buf *buf);
|
|
void bt_hci_extended_inquiry_result(struct net_buf *buf);
|
|
void bt_hci_remote_name_request_complete(struct net_buf *buf);
|
|
|
|
void bt_hci_read_remote_features_complete(struct net_buf *buf);
|
|
void bt_hci_read_remote_ext_features_complete(struct net_buf *buf);
|
|
void bt_hci_role_change(struct net_buf *buf);
|
|
void bt_hci_synchronous_conn_complete(struct net_buf *buf);
|
|
|
|
void bt_hci_le_df_connection_iq_report(struct net_buf *buf);
|
|
void bt_hci_le_df_cte_req_failed(struct net_buf *buf);
|
|
|
|
/** First thread that called bt_recv. NULL if not yet called.
|
|
* Updated non-atomically; may be used only to compare to current thread id.
|
|
*/
|
|
extern k_tid_t bt_recv_thread_id;
|