mirror of
https://github.com/esphome/esphome.git
synced 2025-09-05 04:42:21 +01:00
653 lines
20 KiB
C++
653 lines
20 KiB
C++
#include "ble.h"
|
|
|
|
#ifdef USE_ESP32
|
|
|
|
#include "esphome/core/application.h"
|
|
#include "esphome/core/helpers.h"
|
|
#include "esphome/core/log.h"
|
|
|
|
#include <esp_bt.h>
|
|
#include <esp_bt_device.h>
|
|
#include <esp_bt_main.h>
|
|
#include <esp_gap_ble_api.h>
|
|
#include <freertos/FreeRTOS.h>
|
|
#include <freertos/FreeRTOSConfig.h>
|
|
#include <freertos/task.h>
|
|
#include <nvs_flash.h>
|
|
|
|
#ifdef USE_ARDUINO
|
|
#include <esp32-hal-bt.h>
|
|
#endif
|
|
|
|
namespace esphome::esp32_ble {
|
|
|
|
static const char *const TAG = "esp32_ble";
|
|
|
|
void ESP32BLE::setup() {
|
|
global_ble = this;
|
|
if (!ble_pre_setup_()) {
|
|
ESP_LOGE(TAG, "BLE could not be prepared for configuration");
|
|
this->mark_failed();
|
|
return;
|
|
}
|
|
|
|
this->state_ = BLE_COMPONENT_STATE_DISABLED;
|
|
if (this->enable_on_boot_) {
|
|
this->enable();
|
|
}
|
|
}
|
|
|
|
void ESP32BLE::enable() {
|
|
if (this->state_ != BLE_COMPONENT_STATE_DISABLED)
|
|
return;
|
|
|
|
this->state_ = BLE_COMPONENT_STATE_ENABLE;
|
|
}
|
|
|
|
void ESP32BLE::disable() {
|
|
if (this->state_ == BLE_COMPONENT_STATE_DISABLED)
|
|
return;
|
|
|
|
this->state_ = BLE_COMPONENT_STATE_DISABLE;
|
|
}
|
|
|
|
bool ESP32BLE::is_active() { return this->state_ == BLE_COMPONENT_STATE_ACTIVE; }
|
|
|
|
#ifdef USE_ESP32_BLE_ADVERTISING
|
|
void ESP32BLE::advertising_start() {
|
|
this->advertising_init_();
|
|
if (!this->is_active())
|
|
return;
|
|
this->advertising_->start();
|
|
}
|
|
|
|
void ESP32BLE::advertising_set_service_data(const std::vector<uint8_t> &data) {
|
|
this->advertising_init_();
|
|
this->advertising_->set_service_data(data);
|
|
this->advertising_start();
|
|
}
|
|
|
|
void ESP32BLE::advertising_set_manufacturer_data(const std::vector<uint8_t> &data) {
|
|
this->advertising_init_();
|
|
this->advertising_->set_manufacturer_data(data);
|
|
this->advertising_start();
|
|
}
|
|
|
|
void ESP32BLE::advertising_register_raw_advertisement_callback(std::function<void(bool)> &&callback) {
|
|
this->advertising_init_();
|
|
this->advertising_->register_raw_advertisement_callback(std::move(callback));
|
|
}
|
|
|
|
void ESP32BLE::advertising_add_service_uuid(ESPBTUUID uuid) {
|
|
this->advertising_init_();
|
|
this->advertising_->add_service_uuid(uuid);
|
|
this->advertising_start();
|
|
}
|
|
|
|
void ESP32BLE::advertising_remove_service_uuid(ESPBTUUID uuid) {
|
|
this->advertising_init_();
|
|
this->advertising_->remove_service_uuid(uuid);
|
|
this->advertising_start();
|
|
}
|
|
#endif
|
|
|
|
bool ESP32BLE::ble_pre_setup_() {
|
|
esp_err_t err = nvs_flash_init();
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "nvs_flash_init failed: %d", err);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
#ifdef USE_ESP32_BLE_ADVERTISING
|
|
void ESP32BLE::advertising_init_() {
|
|
if (this->advertising_ != nullptr)
|
|
return;
|
|
this->advertising_ = new BLEAdvertising(this->advertising_cycle_time_); // NOLINT(cppcoreguidelines-owning-memory)
|
|
|
|
this->advertising_->set_scan_response(true);
|
|
this->advertising_->set_min_preferred_interval(0x06);
|
|
this->advertising_->set_appearance(this->appearance_);
|
|
}
|
|
#endif
|
|
|
|
bool ESP32BLE::ble_setup_() {
|
|
esp_err_t err;
|
|
#ifdef USE_ARDUINO
|
|
if (!btStart()) {
|
|
ESP_LOGE(TAG, "btStart failed: %d", esp_bt_controller_get_status());
|
|
return false;
|
|
}
|
|
#else
|
|
if (esp_bt_controller_get_status() != ESP_BT_CONTROLLER_STATUS_ENABLED) {
|
|
// start bt controller
|
|
if (esp_bt_controller_get_status() == ESP_BT_CONTROLLER_STATUS_IDLE) {
|
|
esp_bt_controller_config_t cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT();
|
|
err = esp_bt_controller_init(&cfg);
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_bt_controller_init failed: %s", esp_err_to_name(err));
|
|
return false;
|
|
}
|
|
while (esp_bt_controller_get_status() == ESP_BT_CONTROLLER_STATUS_IDLE)
|
|
;
|
|
}
|
|
if (esp_bt_controller_get_status() == ESP_BT_CONTROLLER_STATUS_INITED) {
|
|
err = esp_bt_controller_enable(ESP_BT_MODE_BLE);
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_bt_controller_enable failed: %s", esp_err_to_name(err));
|
|
return false;
|
|
}
|
|
}
|
|
if (esp_bt_controller_get_status() != ESP_BT_CONTROLLER_STATUS_ENABLED) {
|
|
ESP_LOGE(TAG, "esp bt controller enable failed");
|
|
return false;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
esp_bt_controller_mem_release(ESP_BT_MODE_CLASSIC_BT);
|
|
|
|
err = esp_bluedroid_init();
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_bluedroid_init failed: %d", err);
|
|
return false;
|
|
}
|
|
err = esp_bluedroid_enable();
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_bluedroid_enable failed: %d", err);
|
|
return false;
|
|
}
|
|
|
|
if (!this->gap_event_handlers_.empty()) {
|
|
err = esp_ble_gap_register_callback(ESP32BLE::gap_event_handler);
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_ble_gap_register_callback failed: %d", err);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (!this->gatts_event_handlers_.empty()) {
|
|
err = esp_ble_gatts_register_callback(ESP32BLE::gatts_event_handler);
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_ble_gatts_register_callback failed: %d", err);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (!this->gattc_event_handlers_.empty()) {
|
|
err = esp_ble_gattc_register_callback(ESP32BLE::gattc_event_handler);
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_ble_gattc_register_callback failed: %d", err);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
std::string name;
|
|
if (this->name_.has_value()) {
|
|
name = this->name_.value();
|
|
if (App.is_name_add_mac_suffix_enabled()) {
|
|
name += "-" + get_mac_address().substr(6);
|
|
}
|
|
} else {
|
|
name = App.get_name();
|
|
if (name.length() > 20) {
|
|
if (App.is_name_add_mac_suffix_enabled()) {
|
|
name.erase(name.begin() + 13, name.end() - 7); // Remove characters between 13 and the mac address
|
|
} else {
|
|
name = name.substr(0, 20);
|
|
}
|
|
}
|
|
}
|
|
|
|
err = esp_ble_gap_set_device_name(name.c_str());
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_ble_gap_set_device_name failed: %d", err);
|
|
return false;
|
|
}
|
|
|
|
err = esp_ble_gap_set_security_param(ESP_BLE_SM_IOCAP_MODE, &(this->io_cap_), sizeof(uint8_t));
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_ble_gap_set_security_param failed: %d", err);
|
|
return false;
|
|
}
|
|
|
|
// Set TX power for all BLE operations (advertising, scanning, connections)
|
|
err = esp_ble_tx_power_set(ESP_BLE_PWR_TYPE_DEFAULT, this->tx_power_);
|
|
if (err != ESP_OK) {
|
|
ESP_LOGW(TAG, "esp_ble_tx_power_set failed: %s", esp_err_to_name(err));
|
|
// Continue anyway as this is not critical
|
|
} else {
|
|
ESP_LOGD(TAG, "BLE TX power set to level %d", this->tx_power_);
|
|
}
|
|
|
|
// BLE takes some time to be fully set up, 200ms should be more than enough
|
|
delay(200); // NOLINT
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ESP32BLE::ble_dismantle_() {
|
|
esp_err_t err = esp_bluedroid_disable();
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_bluedroid_disable failed: %d", err);
|
|
return false;
|
|
}
|
|
err = esp_bluedroid_deinit();
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_bluedroid_deinit failed: %d", err);
|
|
return false;
|
|
}
|
|
|
|
#ifdef USE_ARDUINO
|
|
if (!btStop()) {
|
|
ESP_LOGE(TAG, "btStop failed: %d", esp_bt_controller_get_status());
|
|
return false;
|
|
}
|
|
#else
|
|
if (esp_bt_controller_get_status() != ESP_BT_CONTROLLER_STATUS_IDLE) {
|
|
// stop bt controller
|
|
if (esp_bt_controller_get_status() == ESP_BT_CONTROLLER_STATUS_ENABLED) {
|
|
err = esp_bt_controller_disable();
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_bt_controller_disable failed: %s", esp_err_to_name(err));
|
|
return false;
|
|
}
|
|
while (esp_bt_controller_get_status() == ESP_BT_CONTROLLER_STATUS_ENABLED)
|
|
;
|
|
}
|
|
if (esp_bt_controller_get_status() == ESP_BT_CONTROLLER_STATUS_INITED) {
|
|
err = esp_bt_controller_deinit();
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "esp_bt_controller_deinit failed: %s", esp_err_to_name(err));
|
|
return false;
|
|
}
|
|
}
|
|
if (esp_bt_controller_get_status() != ESP_BT_CONTROLLER_STATUS_IDLE) {
|
|
ESP_LOGE(TAG, "esp bt controller disable failed");
|
|
return false;
|
|
}
|
|
}
|
|
#endif
|
|
return true;
|
|
}
|
|
|
|
void ESP32BLE::loop() {
|
|
switch (this->state_) {
|
|
case BLE_COMPONENT_STATE_OFF:
|
|
case BLE_COMPONENT_STATE_DISABLED:
|
|
return;
|
|
case BLE_COMPONENT_STATE_DISABLE: {
|
|
ESP_LOGD(TAG, "Disabling");
|
|
|
|
for (auto *ble_event_handler : this->ble_status_event_handlers_) {
|
|
ble_event_handler->ble_before_disabled_event_handler();
|
|
}
|
|
|
|
if (!ble_dismantle_()) {
|
|
ESP_LOGE(TAG, "Could not be dismantled");
|
|
this->mark_failed();
|
|
return;
|
|
}
|
|
this->state_ = BLE_COMPONENT_STATE_DISABLED;
|
|
return;
|
|
}
|
|
case BLE_COMPONENT_STATE_ENABLE: {
|
|
ESP_LOGD(TAG, "Enabling");
|
|
this->state_ = BLE_COMPONENT_STATE_OFF;
|
|
|
|
if (!ble_setup_()) {
|
|
ESP_LOGE(TAG, "Could not be set up");
|
|
this->mark_failed();
|
|
return;
|
|
}
|
|
|
|
this->state_ = BLE_COMPONENT_STATE_ACTIVE;
|
|
return;
|
|
}
|
|
case BLE_COMPONENT_STATE_ACTIVE:
|
|
break;
|
|
}
|
|
|
|
BLEEvent *ble_event = this->ble_events_.pop();
|
|
while (ble_event != nullptr) {
|
|
switch (ble_event->type_) {
|
|
case BLEEvent::GATTS: {
|
|
esp_gatts_cb_event_t event = ble_event->event_.gatts.gatts_event;
|
|
esp_gatt_if_t gatts_if = ble_event->event_.gatts.gatts_if;
|
|
esp_ble_gatts_cb_param_t *param = &ble_event->event_.gatts.gatts_param;
|
|
ESP_LOGV(TAG, "gatts_event [esp_gatt_if: %d] - %d", gatts_if, event);
|
|
for (auto *gatts_handler : this->gatts_event_handlers_) {
|
|
gatts_handler->gatts_event_handler(event, gatts_if, param);
|
|
}
|
|
break;
|
|
}
|
|
case BLEEvent::GATTC: {
|
|
esp_gattc_cb_event_t event = ble_event->event_.gattc.gattc_event;
|
|
esp_gatt_if_t gattc_if = ble_event->event_.gattc.gattc_if;
|
|
esp_ble_gattc_cb_param_t *param = &ble_event->event_.gattc.gattc_param;
|
|
ESP_LOGV(TAG, "gattc_event [esp_gatt_if: %d] - %d", gattc_if, event);
|
|
for (auto *gattc_handler : this->gattc_event_handlers_) {
|
|
gattc_handler->gattc_event_handler(event, gattc_if, param);
|
|
}
|
|
break;
|
|
}
|
|
case BLEEvent::GAP: {
|
|
esp_gap_ble_cb_event_t gap_event = ble_event->event_.gap.gap_event;
|
|
switch (gap_event) {
|
|
case ESP_GAP_BLE_SCAN_RESULT_EVT:
|
|
// Use the new scan event handler - no memcpy!
|
|
for (auto *scan_handler : this->gap_scan_event_handlers_) {
|
|
scan_handler->gap_scan_event_handler(ble_event->scan_result());
|
|
}
|
|
break;
|
|
|
|
// Scan complete events
|
|
case ESP_GAP_BLE_SCAN_PARAM_SET_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_SCAN_START_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_SCAN_STOP_COMPLETE_EVT:
|
|
// All three scan complete events have the same structure with just status
|
|
// The scan_complete struct matches ESP-IDF's layout exactly, so this reinterpret_cast is safe
|
|
// This is verified at compile-time by static_assert checks in ble_event.h
|
|
// The struct already contains our copy of the status (copied in BLEEvent constructor)
|
|
ESP_LOGV(TAG, "gap_event_handler - %d", gap_event);
|
|
for (auto *gap_handler : this->gap_event_handlers_) {
|
|
gap_handler->gap_event_handler(
|
|
gap_event, reinterpret_cast<esp_ble_gap_cb_param_t *>(&ble_event->event_.gap.scan_complete));
|
|
}
|
|
break;
|
|
|
|
// Advertising complete events
|
|
case ESP_GAP_BLE_ADV_DATA_SET_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_SCAN_RSP_DATA_SET_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_ADV_DATA_RAW_SET_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_ADV_START_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_ADV_STOP_COMPLETE_EVT:
|
|
// All advertising complete events have the same structure with just status
|
|
ESP_LOGV(TAG, "gap_event_handler - %d", gap_event);
|
|
for (auto *gap_handler : this->gap_event_handlers_) {
|
|
gap_handler->gap_event_handler(
|
|
gap_event, reinterpret_cast<esp_ble_gap_cb_param_t *>(&ble_event->event_.gap.adv_complete));
|
|
}
|
|
break;
|
|
|
|
// RSSI complete event
|
|
case ESP_GAP_BLE_READ_RSSI_COMPLETE_EVT:
|
|
ESP_LOGV(TAG, "gap_event_handler - %d", gap_event);
|
|
for (auto *gap_handler : this->gap_event_handlers_) {
|
|
gap_handler->gap_event_handler(
|
|
gap_event, reinterpret_cast<esp_ble_gap_cb_param_t *>(&ble_event->event_.gap.read_rssi_complete));
|
|
}
|
|
break;
|
|
|
|
// Security events
|
|
case ESP_GAP_BLE_AUTH_CMPL_EVT:
|
|
case ESP_GAP_BLE_SEC_REQ_EVT:
|
|
case ESP_GAP_BLE_PASSKEY_NOTIF_EVT:
|
|
case ESP_GAP_BLE_PASSKEY_REQ_EVT:
|
|
case ESP_GAP_BLE_NC_REQ_EVT:
|
|
ESP_LOGV(TAG, "gap_event_handler - %d", gap_event);
|
|
for (auto *gap_handler : this->gap_event_handlers_) {
|
|
gap_handler->gap_event_handler(
|
|
gap_event, reinterpret_cast<esp_ble_gap_cb_param_t *>(&ble_event->event_.gap.security));
|
|
}
|
|
break;
|
|
|
|
default:
|
|
// Unknown/unhandled event
|
|
ESP_LOGW(TAG, "Unhandled GAP event type in loop: %d", gap_event);
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
// Return the event to the pool
|
|
this->ble_event_pool_.release(ble_event);
|
|
ble_event = this->ble_events_.pop();
|
|
}
|
|
#ifdef USE_ESP32_BLE_ADVERTISING
|
|
if (this->advertising_ != nullptr) {
|
|
this->advertising_->loop();
|
|
}
|
|
#endif
|
|
|
|
// Log dropped events periodically
|
|
uint16_t dropped = this->ble_events_.get_and_reset_dropped_count();
|
|
if (dropped > 0) {
|
|
ESP_LOGW(TAG, "Dropped %u BLE events due to buffer overflow", dropped);
|
|
}
|
|
}
|
|
|
|
// Helper function to load new event data based on type
|
|
void load_ble_event(BLEEvent *event, esp_gap_ble_cb_event_t e, esp_ble_gap_cb_param_t *p) {
|
|
event->load_gap_event(e, p);
|
|
}
|
|
|
|
void load_ble_event(BLEEvent *event, esp_gattc_cb_event_t e, esp_gatt_if_t i, esp_ble_gattc_cb_param_t *p) {
|
|
event->load_gattc_event(e, i, p);
|
|
}
|
|
|
|
void load_ble_event(BLEEvent *event, esp_gatts_cb_event_t e, esp_gatt_if_t i, esp_ble_gatts_cb_param_t *p) {
|
|
event->load_gatts_event(e, i, p);
|
|
}
|
|
|
|
template<typename... Args> void enqueue_ble_event(Args... args) {
|
|
// Allocate an event from the pool
|
|
BLEEvent *event = global_ble->ble_event_pool_.allocate();
|
|
if (event == nullptr) {
|
|
// No events available - queue is full or we're out of memory
|
|
global_ble->ble_events_.increment_dropped_count();
|
|
return;
|
|
}
|
|
|
|
// Load new event data (replaces previous event)
|
|
load_ble_event(event, args...);
|
|
|
|
// Push the event to the queue
|
|
global_ble->ble_events_.push(event);
|
|
// Push always succeeds because we're the only producer and the pool ensures we never exceed queue size
|
|
}
|
|
|
|
// Explicit template instantiations for the friend function
|
|
template void enqueue_ble_event(esp_gap_ble_cb_event_t, esp_ble_gap_cb_param_t *);
|
|
template void enqueue_ble_event(esp_gatts_cb_event_t, esp_gatt_if_t, esp_ble_gatts_cb_param_t *);
|
|
template void enqueue_ble_event(esp_gattc_cb_event_t, esp_gatt_if_t, esp_ble_gattc_cb_param_t *);
|
|
|
|
void ESP32BLE::gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) {
|
|
switch (event) {
|
|
// Queue GAP events that components need to handle
|
|
// Scanning events - used by esp32_ble_tracker
|
|
case ESP_GAP_BLE_SCAN_RESULT_EVT:
|
|
case ESP_GAP_BLE_SCAN_PARAM_SET_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_SCAN_START_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_SCAN_STOP_COMPLETE_EVT:
|
|
// Advertising events - used by esp32_ble_beacon and esp32_ble server
|
|
case ESP_GAP_BLE_ADV_DATA_SET_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_SCAN_RSP_DATA_SET_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_ADV_DATA_RAW_SET_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_ADV_START_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_ADV_STOP_COMPLETE_EVT:
|
|
// Connection events - used by ble_client
|
|
case ESP_GAP_BLE_READ_RSSI_COMPLETE_EVT:
|
|
// Security events - used by ble_client and bluetooth_proxy
|
|
case ESP_GAP_BLE_AUTH_CMPL_EVT:
|
|
case ESP_GAP_BLE_SEC_REQ_EVT:
|
|
case ESP_GAP_BLE_PASSKEY_NOTIF_EVT:
|
|
case ESP_GAP_BLE_PASSKEY_REQ_EVT:
|
|
case ESP_GAP_BLE_NC_REQ_EVT:
|
|
enqueue_ble_event(event, param);
|
|
return;
|
|
|
|
// Ignore these GAP events as they are not relevant for our use case
|
|
case ESP_GAP_BLE_UPDATE_CONN_PARAMS_EVT:
|
|
case ESP_GAP_BLE_SET_PKT_LENGTH_COMPLETE_EVT:
|
|
case ESP_GAP_BLE_PHY_UPDATE_COMPLETE_EVT: // BLE 5.0 PHY update complete
|
|
case ESP_GAP_BLE_CHANNEL_SELECT_ALGORITHM_EVT: // BLE 5.0 channel selection algorithm
|
|
return;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
ESP_LOGW(TAG, "Ignoring unexpected GAP event type: %d", event);
|
|
}
|
|
|
|
void ESP32BLE::gatts_event_handler(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if,
|
|
esp_ble_gatts_cb_param_t *param) {
|
|
enqueue_ble_event(event, gatts_if, param);
|
|
}
|
|
|
|
void ESP32BLE::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
|
esp_ble_gattc_cb_param_t *param) {
|
|
enqueue_ble_event(event, gattc_if, param);
|
|
}
|
|
|
|
float ESP32BLE::get_setup_priority() const { return setup_priority::BLUETOOTH; }
|
|
|
|
void ESP32BLE::dump_config() {
|
|
const uint8_t *mac_address = esp_bt_dev_get_address();
|
|
if (mac_address) {
|
|
const char *io_capability_s;
|
|
switch (this->io_cap_) {
|
|
case ESP_IO_CAP_OUT:
|
|
io_capability_s = "display_only";
|
|
break;
|
|
case ESP_IO_CAP_IO:
|
|
io_capability_s = "display_yes_no";
|
|
break;
|
|
case ESP_IO_CAP_IN:
|
|
io_capability_s = "keyboard_only";
|
|
break;
|
|
case ESP_IO_CAP_NONE:
|
|
io_capability_s = "none";
|
|
break;
|
|
case ESP_IO_CAP_KBDISP:
|
|
io_capability_s = "keyboard_display";
|
|
break;
|
|
default:
|
|
io_capability_s = "invalid";
|
|
break;
|
|
}
|
|
// Convert TX power level to dBm for display
|
|
int tx_power_dbm = 0;
|
|
#if defined(CONFIG_IDF_TARGET_ESP32)
|
|
// ESP32 classic power levels (0-7)
|
|
switch (this->tx_power_) {
|
|
case 0:
|
|
tx_power_dbm = -12;
|
|
break; // ESP_PWR_LVL_N12
|
|
case 1:
|
|
tx_power_dbm = -9;
|
|
break; // ESP_PWR_LVL_N9
|
|
case 2:
|
|
tx_power_dbm = -6;
|
|
break; // ESP_PWR_LVL_N6
|
|
case 3:
|
|
tx_power_dbm = -3;
|
|
break; // ESP_PWR_LVL_N3
|
|
case 4:
|
|
tx_power_dbm = 0;
|
|
break; // ESP_PWR_LVL_N0
|
|
case 5:
|
|
tx_power_dbm = 3;
|
|
break; // ESP_PWR_LVL_P3
|
|
case 6:
|
|
tx_power_dbm = 6;
|
|
break; // ESP_PWR_LVL_P6
|
|
case 7:
|
|
tx_power_dbm = 9;
|
|
break; // ESP_PWR_LVL_P9
|
|
default:
|
|
tx_power_dbm = 0;
|
|
break;
|
|
}
|
|
#elif defined(CONFIG_IDF_TARGET_ESP32C2) || defined(CONFIG_IDF_TARGET_ESP32C3) || \
|
|
defined(CONFIG_IDF_TARGET_ESP32C5) || defined(CONFIG_IDF_TARGET_ESP32C6) || defined(CONFIG_IDF_TARGET_ESP32H2) || \
|
|
defined(CONFIG_IDF_TARGET_ESP32S3)
|
|
// Extended power levels for C2/C3/C5/C6/H2/S3 (0-15)
|
|
switch (this->tx_power_) {
|
|
case 0:
|
|
tx_power_dbm = -24;
|
|
break; // ESP_PWR_LVL_N24
|
|
case 1:
|
|
tx_power_dbm = -21;
|
|
break; // ESP_PWR_LVL_N21
|
|
case 2:
|
|
tx_power_dbm = -18;
|
|
break; // ESP_PWR_LVL_N18
|
|
case 3:
|
|
tx_power_dbm = -15;
|
|
break; // ESP_PWR_LVL_N15
|
|
case 4:
|
|
tx_power_dbm = -12;
|
|
break; // ESP_PWR_LVL_N12
|
|
case 5:
|
|
tx_power_dbm = -9;
|
|
break; // ESP_PWR_LVL_N9
|
|
case 6:
|
|
tx_power_dbm = -6;
|
|
break; // ESP_PWR_LVL_N6
|
|
case 7:
|
|
tx_power_dbm = -3;
|
|
break; // ESP_PWR_LVL_N3
|
|
case 8:
|
|
tx_power_dbm = 0;
|
|
break; // ESP_PWR_LVL_N0
|
|
case 9:
|
|
tx_power_dbm = 3;
|
|
break; // ESP_PWR_LVL_P3
|
|
case 10:
|
|
tx_power_dbm = 6;
|
|
break; // ESP_PWR_LVL_P6
|
|
case 11:
|
|
tx_power_dbm = 9;
|
|
break; // ESP_PWR_LVL_P9
|
|
case 12:
|
|
tx_power_dbm = 12;
|
|
break; // ESP_PWR_LVL_P12
|
|
case 13:
|
|
tx_power_dbm = 15;
|
|
break; // ESP_PWR_LVL_P15
|
|
case 14:
|
|
tx_power_dbm = 18;
|
|
break; // ESP_PWR_LVL_P18
|
|
case 15:
|
|
tx_power_dbm = 20;
|
|
break; // ESP_PWR_LVL_P20
|
|
default:
|
|
tx_power_dbm = 0;
|
|
break;
|
|
}
|
|
#else
|
|
// Unknown variant
|
|
tx_power_dbm = 0;
|
|
#endif
|
|
ESP_LOGCONFIG(TAG,
|
|
"BLE:\n"
|
|
" MAC address: %s\n"
|
|
" IO Capability: %s\n"
|
|
" TX Power: %d dBm",
|
|
format_mac_address_pretty(mac_address).c_str(), io_capability_s, tx_power_dbm);
|
|
} else {
|
|
ESP_LOGCONFIG(TAG, "Bluetooth stack is not enabled");
|
|
}
|
|
}
|
|
|
|
uint64_t ble_addr_to_uint64(const esp_bd_addr_t address) {
|
|
uint64_t u = 0;
|
|
u |= uint64_t(address[0] & 0xFF) << 40;
|
|
u |= uint64_t(address[1] & 0xFF) << 32;
|
|
u |= uint64_t(address[2] & 0xFF) << 24;
|
|
u |= uint64_t(address[3] & 0xFF) << 16;
|
|
u |= uint64_t(address[4] & 0xFF) << 8;
|
|
u |= uint64_t(address[5] & 0xFF) << 0;
|
|
return u;
|
|
}
|
|
|
|
ESP32BLE *global_ble = nullptr; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
|
|
|
|
} // namespace esphome::esp32_ble
|
|
|
|
#endif
|