zephyr/subsys/bluetooth/host/hci_core.h

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;