mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-25 13:13:48 +01:00 
			
		
		
		
	Allow multiple bluetooth proxy connections (#3971)
This commit is contained in:
		| @@ -75,7 +75,7 @@ class APIServer : public Component, public Controller { | |||||||
|   void send_homeassistant_service_call(const HomeassistantServiceResponse &call); |   void send_homeassistant_service_call(const HomeassistantServiceResponse &call); | ||||||
| #ifdef USE_BLUETOOTH_PROXY | #ifdef USE_BLUETOOTH_PROXY | ||||||
|   void send_bluetooth_le_advertisement(const BluetoothLEAdvertisementResponse &call); |   void send_bluetooth_le_advertisement(const BluetoothLEAdvertisementResponse &call); | ||||||
|   void send_bluetooth_device_connection(uint64_t address, bool connected, uint16_t mtu, esp_err_t error = ESP_OK); |   void send_bluetooth_device_connection(uint64_t address, bool connected, uint16_t mtu = 0, esp_err_t error = ESP_OK); | ||||||
|   void send_bluetooth_connections_free(uint8_t free, uint8_t limit); |   void send_bluetooth_connections_free(uint8_t free, uint8_t limit); | ||||||
|   void send_bluetooth_gatt_read_response(const BluetoothGATTReadResponse &call); |   void send_bluetooth_gatt_read_response(const BluetoothGATTReadResponse &call); | ||||||
|   void send_bluetooth_gatt_write_response(const BluetoothGATTWriteResponse &call); |   void send_bluetooth_gatt_write_response(const BluetoothGATTWriteResponse &call); | ||||||
|   | |||||||
| @@ -47,11 +47,12 @@ void BLEClient::set_enabled(bool enabled) { | |||||||
|   this->enabled = enabled; |   this->enabled = enabled; | ||||||
| } | } | ||||||
|  |  | ||||||
| void BLEClient::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t esp_gattc_if, | bool BLEClient::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t esp_gattc_if, | ||||||
|                                     esp_ble_gattc_cb_param_t *param) { |                                     esp_ble_gattc_cb_param_t *param) { | ||||||
|   bool all_established = this->all_nodes_established_(); |   bool all_established = this->all_nodes_established_(); | ||||||
|  |  | ||||||
|   BLEClientBase::gattc_event_handler(event, esp_gattc_if, param); |   if (!BLEClientBase::gattc_event_handler(event, esp_gattc_if, param)) | ||||||
|  |     return false; | ||||||
|  |  | ||||||
|   for (auto *node : this->nodes_) |   for (auto *node : this->nodes_) | ||||||
|     node->gattc_event_handler(event, esp_gattc_if, param); |     node->gattc_event_handler(event, esp_gattc_if, param); | ||||||
| @@ -62,6 +63,7 @@ void BLEClient::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t es | |||||||
|       delete svc;  // NOLINT(cppcoreguidelines-owning-memory) |       delete svc;  // NOLINT(cppcoreguidelines-owning-memory) | ||||||
|     this->services_.clear(); |     this->services_.clear(); | ||||||
|   } |   } | ||||||
|  |   return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| void BLEClient::gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) { | void BLEClient::gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) { | ||||||
|   | |||||||
| @@ -50,7 +50,7 @@ class BLEClient : public BLEClientBase { | |||||||
|   void dump_config() override; |   void dump_config() override; | ||||||
|   void loop() override; |   void loop() override; | ||||||
|  |  | ||||||
|   void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, |   bool gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, | ||||||
|                            esp_ble_gattc_cb_param_t *param) override; |                            esp_ble_gattc_cb_param_t *param) override; | ||||||
|  |  | ||||||
|   void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) override; |   void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) override; | ||||||
|   | |||||||
| @@ -7,19 +7,56 @@ AUTO_LOAD = ["esp32_ble_client", "esp32_ble_tracker"] | |||||||
| DEPENDENCIES = ["api", "esp32"] | DEPENDENCIES = ["api", "esp32"] | ||||||
| CODEOWNERS = ["@jesserockz"] | CODEOWNERS = ["@jesserockz"] | ||||||
|  |  | ||||||
|  | CONF_CONNECTIONS = "connections" | ||||||
|  | MAX_CONNECTIONS = 3 | ||||||
|  |  | ||||||
| bluetooth_proxy_ns = cg.esphome_ns.namespace("bluetooth_proxy") | bluetooth_proxy_ns = cg.esphome_ns.namespace("bluetooth_proxy") | ||||||
|  |  | ||||||
| BluetoothProxy = bluetooth_proxy_ns.class_( | BluetoothProxy = bluetooth_proxy_ns.class_( | ||||||
|     "BluetoothProxy", esp32_ble_client.BLEClientBase |     "BluetoothProxy", esp32_ble_tracker.ESPBTDeviceListener, cg.Component | ||||||
|  | ) | ||||||
|  | BluetoothConnection = bluetooth_proxy_ns.class_( | ||||||
|  |     "BluetoothConnection", esp32_ble_client.BLEClientBase | ||||||
| ) | ) | ||||||
|  |  | ||||||
| CONFIG_SCHEMA = cv.Schema( | CONNECTION_SCHEMA = esp32_ble_tracker.ESP_BLE_DEVICE_SCHEMA.extend( | ||||||
|     { |     { | ||||||
|         cv.GenerateID(): cv.declare_id(BluetoothProxy), |         cv.GenerateID(): cv.declare_id(BluetoothConnection), | ||||||
|         cv.Optional(CONF_ACTIVE, default=False): cv.boolean, |  | ||||||
|     } |     } | ||||||
| ).extend(esp32_ble_tracker.ESP_BLE_DEVICE_SCHEMA) | ).extend(cv.COMPONENT_SCHEMA) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def validate_connections(config): | ||||||
|  |     if CONF_CONNECTIONS in config: | ||||||
|  |         if not config[CONF_ACTIVE]: | ||||||
|  |             raise cv.Invalid( | ||||||
|  |                 "Connections can only be used if the proxy is set to active" | ||||||
|  |             ) | ||||||
|  |     else: | ||||||
|  |         if config[CONF_ACTIVE]: | ||||||
|  |             conf = config.copy() | ||||||
|  |             conf[CONF_CONNECTIONS] = [ | ||||||
|  |                 CONNECTION_SCHEMA({}) for _ in range(MAX_CONNECTIONS) | ||||||
|  |             ] | ||||||
|  |             return conf | ||||||
|  |     return config | ||||||
|  |  | ||||||
|  |  | ||||||
|  | CONFIG_SCHEMA = cv.All( | ||||||
|  |     cv.Schema( | ||||||
|  |         { | ||||||
|  |             cv.GenerateID(): cv.declare_id(BluetoothProxy), | ||||||
|  |             cv.Optional(CONF_ACTIVE, default=False): cv.boolean, | ||||||
|  |             cv.Optional(CONF_CONNECTIONS): cv.All( | ||||||
|  |                 cv.ensure_list(CONNECTION_SCHEMA), | ||||||
|  |                 cv.Length(min=1, max=MAX_CONNECTIONS), | ||||||
|  |             ), | ||||||
|  |         } | ||||||
|  |     ) | ||||||
|  |     .extend(esp32_ble_tracker.ESP_BLE_DEVICE_SCHEMA) | ||||||
|  |     .extend(cv.COMPONENT_SCHEMA), | ||||||
|  |     validate_connections, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
| async def to_code(config): | async def to_code(config): | ||||||
| @@ -27,7 +64,12 @@ async def to_code(config): | |||||||
|     await cg.register_component(var, config) |     await cg.register_component(var, config) | ||||||
|  |  | ||||||
|     cg.add(var.set_active(config[CONF_ACTIVE])) |     cg.add(var.set_active(config[CONF_ACTIVE])) | ||||||
|  |     await esp32_ble_tracker.register_ble_device(var, config) | ||||||
|  |  | ||||||
|     await esp32_ble_tracker.register_client(var, config) |     for connection_conf in config.get(CONF_CONNECTIONS, []): | ||||||
|  |         connection_var = cg.new_Pvariable(connection_conf[CONF_ID]) | ||||||
|  |         await cg.register_component(connection_var, connection_conf) | ||||||
|  |         cg.add(var.register_connection(connection_var)) | ||||||
|  |         await esp32_ble_tracker.register_client(connection_var, connection_conf) | ||||||
|  |  | ||||||
|     cg.add_define("USE_BLUETOOTH_PROXY") |     cg.add_define("USE_BLUETOOTH_PROXY") | ||||||
|   | |||||||
							
								
								
									
										281
									
								
								esphome/components/bluetooth_proxy/bluetooth_connection.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										281
									
								
								esphome/components/bluetooth_proxy/bluetooth_connection.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,281 @@ | |||||||
|  | #include "bluetooth_connection.h" | ||||||
|  |  | ||||||
|  | #include "esphome/components/api/api_server.h" | ||||||
|  | #include "esphome/core/helpers.h" | ||||||
|  | #include "esphome/core/log.h" | ||||||
|  |  | ||||||
|  | #ifdef USE_ESP32 | ||||||
|  |  | ||||||
|  | #include "bluetooth_proxy.h" | ||||||
|  |  | ||||||
|  | namespace esphome { | ||||||
|  | namespace bluetooth_proxy { | ||||||
|  |  | ||||||
|  | static const char *const TAG = "bluetooth_proxy.connection"; | ||||||
|  |  | ||||||
|  | bool BluetoothConnection::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, | ||||||
|  |                                               esp_ble_gattc_cb_param_t *param) { | ||||||
|  |   if (!BLEClientBase::gattc_event_handler(event, gattc_if, param)) | ||||||
|  |     return false; | ||||||
|  |  | ||||||
|  |   switch (event) { | ||||||
|  |     case ESP_GATTC_DISCONNECT_EVT: { | ||||||
|  |       api::global_api_server->send_bluetooth_device_connection(this->address_, false, 0, param->disconnect.reason); | ||||||
|  |       this->set_address(0); | ||||||
|  |       api::global_api_server->send_bluetooth_connections_free(this->proxy_->get_bluetooth_connections_free(), | ||||||
|  |                                                               this->proxy_->get_bluetooth_connections_limit()); | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     case ESP_GATTC_OPEN_EVT: { | ||||||
|  |       if (param->open.conn_id != this->conn_id_) | ||||||
|  |         break; | ||||||
|  |       if (param->open.status != ESP_GATT_OK && param->open.status != ESP_GATT_ALREADY_OPEN) { | ||||||
|  |         api::global_api_server->send_bluetooth_device_connection(this->address_, false, 0, param->open.status); | ||||||
|  |         this->set_address(0); | ||||||
|  |         api::global_api_server->send_bluetooth_connections_free(this->proxy_->get_bluetooth_connections_free(), | ||||||
|  |                                                                 this->proxy_->get_bluetooth_connections_limit()); | ||||||
|  |       } | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     case ESP_GATTC_SEARCH_CMPL_EVT: { | ||||||
|  |       if (param->search_cmpl.conn_id != this->conn_id_) | ||||||
|  |         break; | ||||||
|  |       api::global_api_server->send_bluetooth_device_connection(this->address_, true, this->mtu_); | ||||||
|  |       api::global_api_server->send_bluetooth_connections_free(this->proxy_->get_bluetooth_connections_free(), | ||||||
|  |                                                               this->proxy_->get_bluetooth_connections_limit()); | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     case ESP_GATTC_READ_DESCR_EVT: | ||||||
|  |     case ESP_GATTC_READ_CHAR_EVT: { | ||||||
|  |       if (param->read.conn_id != this->conn_id_) | ||||||
|  |         break; | ||||||
|  |       if (param->read.status != ESP_GATT_OK) { | ||||||
|  |         ESP_LOGW(TAG, "[%d] [%s] Error reading char/descriptor at handle 0x%2X, status=%d", this->connection_index_, | ||||||
|  |                  this->address_str_.c_str(), param->read.handle, param->read.status); | ||||||
|  |         api::global_api_server->send_bluetooth_gatt_error(this->address_, param->read.handle, param->read.status); | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       api::BluetoothGATTReadResponse resp; | ||||||
|  |       resp.address = this->address_; | ||||||
|  |       resp.handle = param->read.handle; | ||||||
|  |       resp.data.reserve(param->read.value_len); | ||||||
|  |       for (uint16_t i = 0; i < param->read.value_len; i++) { | ||||||
|  |         resp.data.push_back(param->read.value[i]); | ||||||
|  |       } | ||||||
|  |       api::global_api_server->send_bluetooth_gatt_read_response(resp); | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     case ESP_GATTC_WRITE_CHAR_EVT: | ||||||
|  |     case ESP_GATTC_WRITE_DESCR_EVT: { | ||||||
|  |       if (param->write.conn_id != this->conn_id_) | ||||||
|  |         break; | ||||||
|  |       if (param->write.status != ESP_GATT_OK) { | ||||||
|  |         ESP_LOGW(TAG, "[%d] [%s] Error writing char/descriptor at handle 0x%2X, status=%d", this->connection_index_, | ||||||
|  |                  this->address_str_.c_str(), param->write.handle, param->write.status); | ||||||
|  |         api::global_api_server->send_bluetooth_gatt_error(this->address_, param->write.handle, param->write.status); | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       api::BluetoothGATTWriteResponse resp; | ||||||
|  |       resp.address = this->address_; | ||||||
|  |       resp.handle = param->write.handle; | ||||||
|  |       api::global_api_server->send_bluetooth_gatt_write_response(resp); | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     case ESP_GATTC_UNREG_FOR_NOTIFY_EVT: { | ||||||
|  |       if (this->get_characteristic(param->unreg_for_notify.handle) == nullptr)  // No conn_id in this event | ||||||
|  |         break; | ||||||
|  |       if (param->unreg_for_notify.status != ESP_GATT_OK) { | ||||||
|  |         ESP_LOGW(TAG, "[%d] [%s] Error unregistering notifications for handle 0x%2X, status=%d", | ||||||
|  |                  this->connection_index_, this->address_str_.c_str(), param->unreg_for_notify.handle, | ||||||
|  |                  param->unreg_for_notify.status); | ||||||
|  |         api::global_api_server->send_bluetooth_gatt_error(this->address_, param->unreg_for_notify.handle, | ||||||
|  |                                                           param->unreg_for_notify.status); | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       api::BluetoothGATTNotifyResponse resp; | ||||||
|  |       resp.address = this->address_; | ||||||
|  |       resp.handle = param->unreg_for_notify.handle; | ||||||
|  |       api::global_api_server->send_bluetooth_gatt_notify_response(resp); | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     case ESP_GATTC_REG_FOR_NOTIFY_EVT: { | ||||||
|  |       if (this->get_characteristic(param->reg_for_notify.handle) == nullptr)  // No conn_id in this event | ||||||
|  |         break; | ||||||
|  |       if (param->reg_for_notify.status != ESP_GATT_OK) { | ||||||
|  |         ESP_LOGW(TAG, "[%d] [%s] Error registering notifications for handle 0x%2X, status=%d", this->connection_index_, | ||||||
|  |                  this->address_str_.c_str(), param->reg_for_notify.handle, param->reg_for_notify.status); | ||||||
|  |         api::global_api_server->send_bluetooth_gatt_error(this->address_, param->reg_for_notify.handle, | ||||||
|  |                                                           param->reg_for_notify.status); | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       api::BluetoothGATTNotifyResponse resp; | ||||||
|  |       resp.address = this->address_; | ||||||
|  |       resp.handle = param->reg_for_notify.handle; | ||||||
|  |       api::global_api_server->send_bluetooth_gatt_notify_response(resp); | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     case ESP_GATTC_NOTIFY_EVT: { | ||||||
|  |       if (param->notify.conn_id != this->conn_id_) | ||||||
|  |         break; | ||||||
|  |       ESP_LOGV(TAG, "[%d] [%s] ESP_GATTC_NOTIFY_EVT: handle=0x%2X", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |                param->notify.handle); | ||||||
|  |       api::BluetoothGATTNotifyDataResponse resp; | ||||||
|  |       resp.address = this->address_; | ||||||
|  |       resp.handle = param->notify.handle; | ||||||
|  |       resp.data.reserve(param->notify.value_len); | ||||||
|  |       for (uint16_t i = 0; i < param->notify.value_len; i++) { | ||||||
|  |         resp.data.push_back(param->notify.value[i]); | ||||||
|  |       } | ||||||
|  |       api::global_api_server->send_bluetooth_gatt_notify_data_response(resp); | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     default: | ||||||
|  |       break; | ||||||
|  |   } | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | esp_err_t BluetoothConnection::read_characteristic(uint16_t handle) { | ||||||
|  |   if (!this->connected()) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] Cannot read GATT characteristic, not connected.", this->connection_index_, | ||||||
|  |              this->address_str_.c_str()); | ||||||
|  |     return ESP_GATT_NOT_CONNECTED; | ||||||
|  |   } | ||||||
|  |   auto *characteristic = this->get_characteristic(handle); | ||||||
|  |   if (characteristic == nullptr) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] Cannot read GATT characteristic, not found.", this->connection_index_, | ||||||
|  |              this->address_str_.c_str()); | ||||||
|  |     return ESP_GATT_INVALID_HANDLE; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   ESP_LOGV(TAG, "[%d] [%s] Reading GATT characteristic %s", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |            characteristic->uuid.to_string().c_str()); | ||||||
|  |  | ||||||
|  |   esp_err_t err = | ||||||
|  |       esp_ble_gattc_read_char(this->gattc_if_, this->conn_id_, characteristic->handle, ESP_GATT_AUTH_REQ_NONE); | ||||||
|  |   if (err != ERR_OK) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_read_char error, err=%d", this->connection_index_, | ||||||
|  |              this->address_str_.c_str(), err); | ||||||
|  |     return err; | ||||||
|  |   } | ||||||
|  |   return ESP_OK; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | esp_err_t BluetoothConnection::write_characteristic(uint16_t handle, const std::string &data, bool response) { | ||||||
|  |   if (!this->connected()) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] Cannot write GATT characteristic, not connected.", this->connection_index_, | ||||||
|  |              this->address_str_.c_str()); | ||||||
|  |     return ESP_GATT_NOT_CONNECTED; | ||||||
|  |   } | ||||||
|  |   auto *characteristic = this->get_characteristic(handle); | ||||||
|  |   if (characteristic == nullptr) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] Cannot write GATT characteristic, not found.", this->connection_index_, | ||||||
|  |              this->address_str_.c_str()); | ||||||
|  |     return ESP_GATT_INVALID_HANDLE; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   ESP_LOGV(TAG, "[%d] [%s] Writing GATT characteristic %s", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |            characteristic->uuid.to_string().c_str()); | ||||||
|  |  | ||||||
|  |   auto err = characteristic->write_value((uint8_t *) data.data(), data.size(), | ||||||
|  |                                          response ? ESP_GATT_WRITE_TYPE_RSP : ESP_GATT_WRITE_TYPE_NO_RSP); | ||||||
|  |   if (err != ERR_OK) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_write_char error, err=%d", this->connection_index_, | ||||||
|  |              this->address_str_.c_str(), err); | ||||||
|  |     return err; | ||||||
|  |   } | ||||||
|  |   return ESP_OK; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | esp_err_t BluetoothConnection::read_descriptor(uint16_t handle) { | ||||||
|  |   if (!this->connected()) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] Cannot read GATT descriptor, not connected.", this->connection_index_, | ||||||
|  |              this->address_str_.c_str()); | ||||||
|  |     return ESP_GATT_NOT_CONNECTED; | ||||||
|  |   } | ||||||
|  |   auto *descriptor = this->get_descriptor(handle); | ||||||
|  |   if (descriptor == nullptr) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] Cannot read GATT descriptor, not found.", this->connection_index_, | ||||||
|  |              this->address_str_.c_str()); | ||||||
|  |     return ESP_GATT_INVALID_HANDLE; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   ESP_LOGV(TAG, "[%d] [%s] Reading GATT descriptor %s", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |            descriptor->uuid.to_string().c_str()); | ||||||
|  |  | ||||||
|  |   esp_err_t err = | ||||||
|  |       esp_ble_gattc_read_char_descr(this->gattc_if_, this->conn_id_, descriptor->handle, ESP_GATT_AUTH_REQ_NONE); | ||||||
|  |   if (err != ERR_OK) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_read_char_descr error, err=%d", this->connection_index_, | ||||||
|  |              this->address_str_.c_str(), err); | ||||||
|  |     return err; | ||||||
|  |   } | ||||||
|  |   return ESP_OK; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | esp_err_t BluetoothConnection::write_descriptor(uint16_t handle, const std::string &data) { | ||||||
|  |   if (!this->connected()) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] Cannot write GATT descriptor, not connected.", this->connection_index_, | ||||||
|  |              this->address_str_.c_str()); | ||||||
|  |     return ESP_GATT_NOT_CONNECTED; | ||||||
|  |   } | ||||||
|  |   auto *descriptor = this->get_descriptor(handle); | ||||||
|  |   if (descriptor == nullptr) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] Cannot write GATT descriptor, not found.", this->connection_index_, | ||||||
|  |              this->address_str_.c_str()); | ||||||
|  |     return ESP_GATT_INVALID_HANDLE; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   ESP_LOGV(TAG, "[%d] [%s] Writing GATT descriptor %s", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |            descriptor->uuid.to_string().c_str()); | ||||||
|  |  | ||||||
|  |   auto err = | ||||||
|  |       esp_ble_gattc_write_char_descr(this->gattc_if_, this->conn_id_, descriptor->handle, data.size(), | ||||||
|  |                                      (uint8_t *) data.data(), ESP_GATT_WRITE_TYPE_NO_RSP, ESP_GATT_AUTH_REQ_NONE); | ||||||
|  |   if (err != ERR_OK) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_write_char_descr error, err=%d", this->connection_index_, | ||||||
|  |              this->address_str_.c_str(), err); | ||||||
|  |     return err; | ||||||
|  |   } | ||||||
|  |   return ESP_OK; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | esp_err_t BluetoothConnection::notify_characteristic(uint16_t handle, bool enable) { | ||||||
|  |   if (!this->connected()) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] Cannot notify GATT characteristic, not connected.", this->connection_index_, | ||||||
|  |              this->address_str_.c_str()); | ||||||
|  |     return ESP_GATT_NOT_CONNECTED; | ||||||
|  |   } | ||||||
|  |   auto *characteristic = this->get_characteristic(handle); | ||||||
|  |   if (characteristic == nullptr) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] Cannot notify GATT characteristic, not found.", this->connection_index_, | ||||||
|  |              this->address_str_.c_str()); | ||||||
|  |     return ESP_GATT_INVALID_HANDLE; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (enable) { | ||||||
|  |     ESP_LOGV(TAG, "[%d] [%s] Registering for GATT characteristic notifications %s", this->connection_index_, | ||||||
|  |              this->address_str_.c_str(), characteristic->uuid.to_string().c_str()); | ||||||
|  |     esp_err_t err = esp_ble_gattc_register_for_notify(this->gattc_if_, this->remote_bda_, characteristic->handle); | ||||||
|  |     if (err != ESP_OK) { | ||||||
|  |       ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_register_for_notify failed, err=%d", this->connection_index_, | ||||||
|  |                this->address_str_.c_str(), err); | ||||||
|  |       return err; | ||||||
|  |     } | ||||||
|  |   } else { | ||||||
|  |     ESP_LOGV(TAG, "[%d] [%s] Unregistering for GATT characteristic notifications %s", this->connection_index_, | ||||||
|  |              this->address_str_.c_str(), characteristic->uuid.to_string().c_str()); | ||||||
|  |     esp_err_t err = esp_ble_gattc_unregister_for_notify(this->gattc_if_, this->remote_bda_, characteristic->handle); | ||||||
|  |     if (err != ESP_OK) { | ||||||
|  |       ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_unregister_for_notify failed, err=%d", this->connection_index_, | ||||||
|  |                this->address_str_.c_str(), err); | ||||||
|  |       return err; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   return ESP_OK; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | }  // namespace bluetooth_proxy | ||||||
|  | }  // namespace esphome | ||||||
|  |  | ||||||
|  | #endif  // USE_ESP32 | ||||||
							
								
								
									
										34
									
								
								esphome/components/bluetooth_proxy/bluetooth_connection.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								esphome/components/bluetooth_proxy/bluetooth_connection.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | |||||||
|  | #pragma once | ||||||
|  |  | ||||||
|  | #ifdef USE_ESP32 | ||||||
|  |  | ||||||
|  | #include "esphome/components/esp32_ble_client/ble_client_base.h" | ||||||
|  |  | ||||||
|  | namespace esphome { | ||||||
|  | namespace bluetooth_proxy { | ||||||
|  |  | ||||||
|  | class BluetoothProxy; | ||||||
|  |  | ||||||
|  | class BluetoothConnection : public esp32_ble_client::BLEClientBase { | ||||||
|  |  public: | ||||||
|  |   bool gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, | ||||||
|  |                            esp_ble_gattc_cb_param_t *param) override; | ||||||
|  |  | ||||||
|  |   esp_err_t read_characteristic(uint16_t handle); | ||||||
|  |   esp_err_t write_characteristic(uint16_t handle, const std::string &data, bool response); | ||||||
|  |   esp_err_t read_descriptor(uint16_t handle); | ||||||
|  |   esp_err_t write_descriptor(uint16_t handle, const std::string &data); | ||||||
|  |  | ||||||
|  |   esp_err_t notify_characteristic(uint16_t handle, bool enable); | ||||||
|  |  | ||||||
|  |  protected: | ||||||
|  |   friend class BluetoothProxy; | ||||||
|  |  | ||||||
|  |   int16_t send_service_{-1}; | ||||||
|  |   BluetoothProxy *proxy_; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | }  // namespace bluetooth_proxy | ||||||
|  | }  // namespace esphome | ||||||
|  |  | ||||||
|  | #endif  // USE_ESP32 | ||||||
| @@ -11,13 +11,7 @@ namespace bluetooth_proxy { | |||||||
|  |  | ||||||
| static const char *const TAG = "bluetooth_proxy"; | static const char *const TAG = "bluetooth_proxy"; | ||||||
|  |  | ||||||
| static const esp_err_t ESP_GATT_NOT_CONNECTED = -1; | BluetoothProxy::BluetoothProxy() { global_bluetooth_proxy = this; } | ||||||
| static const esp_err_t ESP_GATT_WRONG_ADDRESS = -2; |  | ||||||
|  |  | ||||||
| BluetoothProxy::BluetoothProxy() { |  | ||||||
|   global_bluetooth_proxy = this; |  | ||||||
|   this->address_ = 0; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| bool BluetoothProxy::parse_device(const esp32_ble_tracker::ESPBTDevice &device) { | bool BluetoothProxy::parse_device(const esp32_ble_tracker::ESPBTDevice &device) { | ||||||
|   if (!api::global_api_server->is_connected()) |   if (!api::global_api_server->is_connected()) | ||||||
| @@ -26,10 +20,6 @@ bool BluetoothProxy::parse_device(const esp32_ble_tracker::ESPBTDevice &device) | |||||||
|            device.get_rssi()); |            device.get_rssi()); | ||||||
|   this->send_api_packet_(device); |   this->send_api_packet_(device); | ||||||
|  |  | ||||||
|   if (this->address_ == 0) |  | ||||||
|     return true; |  | ||||||
|  |  | ||||||
|   BLEClientBase::parse_device(device); |  | ||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -57,170 +47,101 @@ void BluetoothProxy::send_api_packet_(const esp32_ble_tracker::ESPBTDevice &devi | |||||||
|   api::global_api_server->send_bluetooth_le_advertisement(resp); |   api::global_api_server->send_bluetooth_le_advertisement(resp); | ||||||
| } | } | ||||||
|  |  | ||||||
| void BluetoothProxy::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, | void BluetoothProxy::dump_config() { | ||||||
|                                          esp_ble_gattc_cb_param_t *param) { |   ESP_LOGCONFIG(TAG, "Bluetooth Proxy:"); | ||||||
|   BLEClientBase::gattc_event_handler(event, gattc_if, param); |   ESP_LOGCONFIG(TAG, "  Active: %s", YESNO(this->active_)); | ||||||
|   switch (event) { | } | ||||||
|     case ESP_GATTC_DISCONNECT_EVT: { |  | ||||||
|       api::global_api_server->send_bluetooth_device_connection(this->address_, false, this->mtu_, | void BluetoothProxy::loop() { | ||||||
|                                                                param->disconnect.reason); |   if (!api::global_api_server->is_connected()) { | ||||||
|       api::global_api_server->send_bluetooth_connections_free(this->get_bluetooth_connections_free(), |     for (auto *connection : this->connections_) { | ||||||
|                                                               this->get_bluetooth_connections_limit()); |       if (connection->get_address() != 0) { | ||||||
|       this->address_ = 0; |         connection->disconnect(); | ||||||
|     } |  | ||||||
|     case ESP_GATTC_OPEN_EVT: { |  | ||||||
|       if (param->open.status != ESP_GATT_OK && param->open.status != ESP_GATT_ALREADY_OPEN) { |  | ||||||
|         api::global_api_server->send_bluetooth_device_connection(this->address_, false, this->mtu_, param->open.status); |  | ||||||
|         break; |  | ||||||
|       } |       } | ||||||
|       break; |  | ||||||
|     } |     } | ||||||
|     case ESP_GATTC_SEARCH_CMPL_EVT: { |     return; | ||||||
|       api::global_api_server->send_bluetooth_device_connection(this->address_, true, this->mtu_); |   } | ||||||
|       api::global_api_server->send_bluetooth_connections_free(this->get_bluetooth_connections_free(), |   for (auto *connection : this->connections_) { | ||||||
|                                                               this->get_bluetooth_connections_limit()); |     if (connection->send_service_ == connection->services_.size()) { | ||||||
|       break; |       connection->send_service_ = -1; | ||||||
|     } |       api::global_api_server->send_bluetooth_gatt_services_done(connection->get_address()); | ||||||
|     case ESP_GATTC_READ_DESCR_EVT: |     } else if (connection->send_service_ >= 0) { | ||||||
|     case ESP_GATTC_READ_CHAR_EVT: { |       auto &service = connection->services_[connection->send_service_]; | ||||||
|       if (param->read.conn_id != this->conn_id_) |       api::BluetoothGATTGetServicesResponse resp; | ||||||
|         break; |       resp.address = connection->get_address(); | ||||||
|       if (param->read.status != ESP_GATT_OK) { |       api::BluetoothGATTService service_resp; | ||||||
|         ESP_LOGW(TAG, "Error reading char/descriptor at handle 0x%2X, status=%d", param->read.handle, |       service_resp.uuid = {service->uuid.get_128bit_high(), service->uuid.get_128bit_low()}; | ||||||
|                  param->read.status); |       service_resp.handle = service->start_handle; | ||||||
|         api::global_api_server->send_bluetooth_gatt_error(this->address_, param->read.handle, param->read.status); |       for (auto &characteristic : service->characteristics) { | ||||||
|         break; |         api::BluetoothGATTCharacteristic characteristic_resp; | ||||||
|  |         characteristic_resp.uuid = {characteristic->uuid.get_128bit_high(), characteristic->uuid.get_128bit_low()}; | ||||||
|  |         characteristic_resp.handle = characteristic->handle; | ||||||
|  |         characteristic_resp.properties = characteristic->properties; | ||||||
|  |         for (auto &descriptor : characteristic->descriptors) { | ||||||
|  |           api::BluetoothGATTDescriptor descriptor_resp; | ||||||
|  |           descriptor_resp.uuid = {descriptor->uuid.get_128bit_high(), descriptor->uuid.get_128bit_low()}; | ||||||
|  |           descriptor_resp.handle = descriptor->handle; | ||||||
|  |           characteristic_resp.descriptors.push_back(std::move(descriptor_resp)); | ||||||
|  |         } | ||||||
|  |         service_resp.characteristics.push_back(std::move(characteristic_resp)); | ||||||
|       } |       } | ||||||
|       api::BluetoothGATTReadResponse resp; |       resp.services.push_back(std::move(service_resp)); | ||||||
|       resp.address = this->address_; |       api::global_api_server->send_bluetooth_gatt_services(resp); | ||||||
|       resp.handle = param->read.handle; |       connection->send_service_++; | ||||||
|       resp.data.reserve(param->read.value_len); |  | ||||||
|       for (uint16_t i = 0; i < param->read.value_len; i++) { |  | ||||||
|         resp.data.push_back(param->read.value[i]); |  | ||||||
|       } |  | ||||||
|       api::global_api_server->send_bluetooth_gatt_read_response(resp); |  | ||||||
|       break; |  | ||||||
|     } |     } | ||||||
|     case ESP_GATTC_WRITE_CHAR_EVT: |  | ||||||
|     case ESP_GATTC_WRITE_DESCR_EVT: { |  | ||||||
|       if (param->write.conn_id != this->conn_id_) |  | ||||||
|         break; |  | ||||||
|       if (param->write.status != ESP_GATT_OK) { |  | ||||||
|         ESP_LOGW(TAG, "Error writing char/descriptor at handle 0x%2X, status=%d", param->write.handle, |  | ||||||
|                  param->write.status); |  | ||||||
|         api::global_api_server->send_bluetooth_gatt_error(this->address_, param->write.handle, param->write.status); |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       api::BluetoothGATTWriteResponse resp; |  | ||||||
|       resp.address = this->address_; |  | ||||||
|       resp.handle = param->write.handle; |  | ||||||
|       api::global_api_server->send_bluetooth_gatt_write_response(resp); |  | ||||||
|       break; |  | ||||||
|     } |  | ||||||
|     case ESP_GATTC_UNREG_FOR_NOTIFY_EVT: { |  | ||||||
|       if (param->unreg_for_notify.status != ESP_GATT_OK) { |  | ||||||
|         ESP_LOGW(TAG, "Error unregistering notifications for handle 0x%2X, status=%d", param->unreg_for_notify.handle, |  | ||||||
|                  param->unreg_for_notify.status); |  | ||||||
|         api::global_api_server->send_bluetooth_gatt_error(this->address_, param->unreg_for_notify.handle, |  | ||||||
|                                                           param->unreg_for_notify.status); |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       api::BluetoothGATTNotifyResponse resp; |  | ||||||
|       resp.address = this->address_; |  | ||||||
|       resp.handle = param->unreg_for_notify.handle; |  | ||||||
|       api::global_api_server->send_bluetooth_gatt_notify_response(resp); |  | ||||||
|       break; |  | ||||||
|     } |  | ||||||
|     case ESP_GATTC_REG_FOR_NOTIFY_EVT: { |  | ||||||
|       if (param->reg_for_notify.status != ESP_GATT_OK) { |  | ||||||
|         ESP_LOGW(TAG, "Error registering notifications for handle 0x%2X, status=%d", param->reg_for_notify.handle, |  | ||||||
|                  param->reg_for_notify.status); |  | ||||||
|         api::global_api_server->send_bluetooth_gatt_error(this->address_, param->reg_for_notify.handle, |  | ||||||
|                                                           param->reg_for_notify.status); |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       api::BluetoothGATTNotifyResponse resp; |  | ||||||
|       resp.address = this->address_; |  | ||||||
|       resp.handle = param->reg_for_notify.handle; |  | ||||||
|       api::global_api_server->send_bluetooth_gatt_notify_response(resp); |  | ||||||
|       break; |  | ||||||
|     } |  | ||||||
|     case ESP_GATTC_NOTIFY_EVT: { |  | ||||||
|       if (param->notify.conn_id != this->conn_id_) |  | ||||||
|         break; |  | ||||||
|       ESP_LOGV(TAG, "ESP_GATTC_NOTIFY_EVT: handle=0x%2X", param->notify.handle); |  | ||||||
|       api::BluetoothGATTNotifyDataResponse resp; |  | ||||||
|       resp.address = this->address_; |  | ||||||
|       resp.handle = param->notify.handle; |  | ||||||
|       resp.data.reserve(param->notify.value_len); |  | ||||||
|       for (uint16_t i = 0; i < param->notify.value_len; i++) { |  | ||||||
|         resp.data.push_back(param->notify.value[i]); |  | ||||||
|       } |  | ||||||
|       api::global_api_server->send_bluetooth_gatt_notify_data_response(resp); |  | ||||||
|       break; |  | ||||||
|     } |  | ||||||
|     default: |  | ||||||
|       break; |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| void BluetoothProxy::dump_config() { ESP_LOGCONFIG(TAG, "Bluetooth Proxy:"); } | BluetoothConnection *BluetoothProxy::get_connection_(uint64_t address, bool reserve) { | ||||||
|  |   for (auto *connection : this->connections_) { | ||||||
|  |     if (connection->get_address() == address) | ||||||
|  |       return connection; | ||||||
|  |   } | ||||||
|  |  | ||||||
| void BluetoothProxy::loop() { |   if (!reserve) | ||||||
|   BLEClientBase::loop(); |     return nullptr; | ||||||
|   if (this->state_ != espbt::ClientState::IDLE && !api::global_api_server->is_connected()) { |  | ||||||
|     ESP_LOGI(TAG, "[%s] Disconnecting.", this->address_str().c_str()); |   for (auto *connection : this->connections_) { | ||||||
|     auto err = esp_ble_gattc_close(this->gattc_if_, this->conn_id_); |     if (connection->get_address() == 0) { | ||||||
|     if (err != ERR_OK) { |       connection->set_address(address); | ||||||
|       ESP_LOGW(TAG, "esp_ble_gattc_close error, address=%s err=%d", this->address_str().c_str(), err); |       return connection; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (this->send_service_ == this->services_.size()) { |   return nullptr; | ||||||
|     this->send_service_ = -1; |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_services_done(this->address_); |  | ||||||
|   } else if (this->send_service_ >= 0) { |  | ||||||
|     auto &service = this->services_[this->send_service_]; |  | ||||||
|     api::BluetoothGATTGetServicesResponse resp; |  | ||||||
|     resp.address = this->address_; |  | ||||||
|     api::BluetoothGATTService service_resp; |  | ||||||
|     service_resp.uuid = {service->uuid.get_128bit_high(), service->uuid.get_128bit_low()}; |  | ||||||
|     service_resp.handle = service->start_handle; |  | ||||||
|     for (auto &characteristic : service->characteristics) { |  | ||||||
|       api::BluetoothGATTCharacteristic characteristic_resp; |  | ||||||
|       characteristic_resp.uuid = {characteristic->uuid.get_128bit_high(), characteristic->uuid.get_128bit_low()}; |  | ||||||
|       characteristic_resp.handle = characteristic->handle; |  | ||||||
|       characteristic_resp.properties = characteristic->properties; |  | ||||||
|       for (auto &descriptor : characteristic->descriptors) { |  | ||||||
|         api::BluetoothGATTDescriptor descriptor_resp; |  | ||||||
|         descriptor_resp.uuid = {descriptor->uuid.get_128bit_high(), descriptor->uuid.get_128bit_low()}; |  | ||||||
|         descriptor_resp.handle = descriptor->handle; |  | ||||||
|         characteristic_resp.descriptors.push_back(std::move(descriptor_resp)); |  | ||||||
|       } |  | ||||||
|       service_resp.characteristics.push_back(std::move(characteristic_resp)); |  | ||||||
|     } |  | ||||||
|     resp.services.push_back(std::move(service_resp)); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_services(resp); |  | ||||||
|     this->send_service_++; |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void BluetoothProxy::bluetooth_device_request(const api::BluetoothDeviceRequest &msg) { | void BluetoothProxy::bluetooth_device_request(const api::BluetoothDeviceRequest &msg) { | ||||||
|   switch (msg.request_type) { |   switch (msg.request_type) { | ||||||
|     case api::enums::BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT: { |     case api::enums::BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT: { | ||||||
|       this->address_ = msg.address; |       auto *connection = this->get_connection_(msg.address, true); | ||||||
|       this->set_state(espbt::ClientState::SEARCHING); |       if (connection == nullptr) { | ||||||
|  |         ESP_LOGW(TAG, "No free connections available"); | ||||||
|  |         api::global_api_server->send_bluetooth_device_connection(msg.address, false); | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       ESP_LOGV(TAG, "[%d] [%s] Searching to connect", connection->get_connection_index(), | ||||||
|  |                connection->address_str().c_str()); | ||||||
|  |       connection->set_state(espbt::ClientState::SEARCHING); | ||||||
|       api::global_api_server->send_bluetooth_connections_free(this->get_bluetooth_connections_free(), |       api::global_api_server->send_bluetooth_connections_free(this->get_bluetooth_connections_free(), | ||||||
|                                                               this->get_bluetooth_connections_limit()); |                                                               this->get_bluetooth_connections_limit()); | ||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
|     case api::enums::BLUETOOTH_DEVICE_REQUEST_TYPE_DISCONNECT: { |     case api::enums::BLUETOOTH_DEVICE_REQUEST_TYPE_DISCONNECT: { | ||||||
|       if (this->state() != espbt::ClientState::IDLE) { |       auto *connection = this->get_connection_(msg.address, false); | ||||||
|         ESP_LOGI(TAG, "[%s] Disconnecting.", this->address_str().c_str()); |       if (connection == nullptr) { | ||||||
|         auto err = esp_ble_gattc_close(this->gattc_if_, this->conn_id_); |         api::global_api_server->send_bluetooth_device_connection(msg.address, false); | ||||||
|         if (err != ERR_OK) { |         api::global_api_server->send_bluetooth_connections_free(this->get_bluetooth_connections_free(), | ||||||
|           ESP_LOGW(TAG, "esp_ble_gattc_close error, address=%s err=%d", this->address_str().c_str(), err); |                                                                 this->get_bluetooth_connections_limit()); | ||||||
|         } |         return; | ||||||
|  |       } | ||||||
|  |       if (connection->state() != espbt::ClientState::IDLE) { | ||||||
|  |         connection->disconnect(); | ||||||
|  |       } else { | ||||||
|  |         connection->set_address(0); | ||||||
|  |         api::global_api_server->send_bluetooth_device_connection(msg.address, false); | ||||||
|  |         api::global_api_server->send_bluetooth_connections_free(this->get_bluetooth_connections_free(), | ||||||
|  |                                                                 this->get_bluetooth_connections_limit()); | ||||||
|       } |       } | ||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
| @@ -231,170 +152,88 @@ void BluetoothProxy::bluetooth_device_request(const api::BluetoothDeviceRequest | |||||||
| } | } | ||||||
|  |  | ||||||
| void BluetoothProxy::bluetooth_gatt_read(const api::BluetoothGATTReadRequest &msg) { | void BluetoothProxy::bluetooth_gatt_read(const api::BluetoothGATTReadRequest &msg) { | ||||||
|   if (this->state_ != espbt::ClientState::ESTABLISHED) { |   auto *connection = this->get_connection_(msg.address, false); | ||||||
|     ESP_LOGW(TAG, "Cannot read GATT characteristic, not connected."); |   if (connection == nullptr) { | ||||||
|  |     ESP_LOGW(TAG, "Cannot read GATT characteristic, not connected"); | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED); |     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED); | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|   if (this->address_ != msg.address) { |  | ||||||
|     ESP_LOGW(TAG, "Address mismatch for read GATT characteristic request"); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_WRONG_ADDRESS); |  | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   auto *characteristic = this->get_characteristic(msg.handle); |   auto err = connection->read_characteristic(msg.handle); | ||||||
|   if (characteristic == nullptr) { |   if (err != ESP_OK) { | ||||||
|     ESP_LOGW(TAG, "Cannot read GATT characteristic, not found."); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_INVALID_HANDLE); |  | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   ESP_LOGV(TAG, "Reading GATT characteristic %s", characteristic->uuid.to_string().c_str()); |  | ||||||
|  |  | ||||||
|   esp_err_t err = |  | ||||||
|       esp_ble_gattc_read_char(this->gattc_if_, this->conn_id_, characteristic->handle, ESP_GATT_AUTH_REQ_NONE); |  | ||||||
|   if (err != ERR_OK) { |  | ||||||
|     ESP_LOGW(TAG, "esp_ble_gattc_read_char error, err=%d", err); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err); |     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| void BluetoothProxy::bluetooth_gatt_write(const api::BluetoothGATTWriteRequest &msg) { | void BluetoothProxy::bluetooth_gatt_write(const api::BluetoothGATTWriteRequest &msg) { | ||||||
|   if (this->state_ != espbt::ClientState::ESTABLISHED) { |   auto *connection = this->get_connection_(msg.address, false); | ||||||
|     ESP_LOGW(TAG, "Cannot write GATT characteristic, not connected."); |   if (connection == nullptr) { | ||||||
|  |     ESP_LOGW(TAG, "Cannot write GATT characteristic, not connected"); | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED); |     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED); | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|   if (this->address_ != msg.address) { |  | ||||||
|     ESP_LOGW(TAG, "Address mismatch for write GATT characteristic request"); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_WRONG_ADDRESS); |  | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   auto *characteristic = this->get_characteristic(msg.handle); |   auto err = connection->write_characteristic(msg.handle, msg.data, msg.response); | ||||||
|   if (characteristic == nullptr) { |   if (err != ESP_OK) { | ||||||
|     ESP_LOGW(TAG, "Cannot write GATT characteristic, not found."); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_INVALID_HANDLE); |  | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   ESP_LOGV(TAG, "Writing GATT characteristic %s", characteristic->uuid.to_string().c_str()); |  | ||||||
|   auto err = characteristic->write_value((uint8_t *) msg.data.data(), msg.data.size(), |  | ||||||
|                                          msg.response ? ESP_GATT_WRITE_TYPE_RSP : ESP_GATT_WRITE_TYPE_NO_RSP); |  | ||||||
|   if (err != ERR_OK) { |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err); |     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| void BluetoothProxy::bluetooth_gatt_read_descriptor(const api::BluetoothGATTReadDescriptorRequest &msg) { | void BluetoothProxy::bluetooth_gatt_read_descriptor(const api::BluetoothGATTReadDescriptorRequest &msg) { | ||||||
|   if (this->state_ != espbt::ClientState::ESTABLISHED) { |   auto *connection = this->get_connection_(msg.address, false); | ||||||
|     ESP_LOGW(TAG, "Cannot read GATT characteristic descriptor, not connected."); |   if (connection == nullptr) { | ||||||
|  |     ESP_LOGW(TAG, "Cannot read GATT descriptor, not connected"); | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED); |     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED); | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|   if (this->address_ != msg.address) { |  | ||||||
|     ESP_LOGW(TAG, "Address mismatch for read GATT characteristic descriptor request"); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_WRONG_ADDRESS); |  | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   auto *descriptor = this->get_descriptor(msg.handle); |   auto err = connection->read_descriptor(msg.handle); | ||||||
|   if (descriptor == nullptr) { |   if (err != ESP_OK) { | ||||||
|     ESP_LOGW(TAG, "Cannot read GATT characteristic descriptor, not found."); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_INVALID_HANDLE); |  | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   ESP_LOGV(TAG, "Reading GATT characteristic descriptor %s -> %s", descriptor->characteristic->uuid.to_string().c_str(), |  | ||||||
|            descriptor->uuid.to_string().c_str()); |  | ||||||
|  |  | ||||||
|   esp_err_t err = |  | ||||||
|       esp_ble_gattc_read_char_descr(this->gattc_if_, this->conn_id_, descriptor->handle, ESP_GATT_AUTH_REQ_NONE); |  | ||||||
|   if (err != ERR_OK) { |  | ||||||
|     ESP_LOGW(TAG, "esp_ble_gattc_read_char error, err=%d", err); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err); |     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| void BluetoothProxy::bluetooth_gatt_write_descriptor(const api::BluetoothGATTWriteDescriptorRequest &msg) { | void BluetoothProxy::bluetooth_gatt_write_descriptor(const api::BluetoothGATTWriteDescriptorRequest &msg) { | ||||||
|   if (this->state_ != espbt::ClientState::ESTABLISHED) { |   auto *connection = this->get_connection_(msg.address, false); | ||||||
|     ESP_LOGW(TAG, "Cannot write GATT characteristic descriptor, not connected."); |   if (connection == nullptr) { | ||||||
|  |     ESP_LOGW(TAG, "Cannot write GATT descriptor, not connected"); | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED); |     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED); | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|   if (this->address_ != msg.address) { |  | ||||||
|     ESP_LOGW(TAG, "Address mismatch for write GATT characteristic descriptor request"); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_WRONG_ADDRESS); |  | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   auto *descriptor = this->get_descriptor(msg.handle); |   auto err = connection->write_descriptor(msg.handle, msg.data); | ||||||
|   if (descriptor == nullptr) { |   if (err != ESP_OK) { | ||||||
|     ESP_LOGW(TAG, "Cannot write GATT characteristic descriptor, not found."); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_INVALID_HANDLE); |  | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   ESP_LOGV(TAG, "Writing GATT characteristic descriptor %s -> %s", descriptor->characteristic->uuid.to_string().c_str(), |  | ||||||
|            descriptor->uuid.to_string().c_str()); |  | ||||||
|  |  | ||||||
|   esp_err_t err = |  | ||||||
|       esp_ble_gattc_write_char_descr(this->gattc_if_, this->conn_id_, descriptor->handle, msg.data.size(), |  | ||||||
|                                      (uint8_t *) msg.data.data(), ESP_GATT_WRITE_TYPE_NO_RSP, ESP_GATT_AUTH_REQ_NONE); |  | ||||||
|   if (err != ERR_OK) { |  | ||||||
|     ESP_LOGW(TAG, "esp_ble_gattc_write_char_descr error, err=%d", err); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err); |     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| void BluetoothProxy::bluetooth_gatt_send_services(const api::BluetoothGATTGetServicesRequest &msg) { | void BluetoothProxy::bluetooth_gatt_send_services(const api::BluetoothGATTGetServicesRequest &msg) { | ||||||
|   if (this->state_ != espbt::ClientState::ESTABLISHED) { |   auto *connection = this->get_connection_(msg.address, false); | ||||||
|     ESP_LOGW(TAG, "Cannot get GATT services, not connected."); |   if (connection == nullptr || !connection->connected()) { | ||||||
|  |     ESP_LOGW(TAG, "Cannot get GATT services, not connected"); | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, 0, ESP_GATT_NOT_CONNECTED); |     api::global_api_server->send_bluetooth_gatt_error(msg.address, 0, ESP_GATT_NOT_CONNECTED); | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|   if (this->address_ != msg.address) { |   if (connection->services_.empty()) { | ||||||
|     ESP_LOGW(TAG, "Address mismatch for service list request"); |     ESP_LOGW(TAG, "[%d] [%s] No GATT services found", connection->connection_index_, connection->address_str().c_str()); | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, 0, ESP_GATT_WRONG_ADDRESS); |     api::global_api_server->send_bluetooth_gatt_services_done(msg.address); | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|   this->send_service_ = 0; |   if (connection->send_service_ == -1)  // Don't start sending services again if we're already sending them | ||||||
|  |     connection->send_service_ = 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| void BluetoothProxy::bluetooth_gatt_notify(const api::BluetoothGATTNotifyRequest &msg) { | void BluetoothProxy::bluetooth_gatt_notify(const api::BluetoothGATTNotifyRequest &msg) { | ||||||
|   if (this->state_ != espbt::ClientState::ESTABLISHED) { |   auto *connection = this->get_connection_(msg.address, false); | ||||||
|     ESP_LOGW(TAG, "Cannot configure notify, not connected."); |   if (connection == nullptr) { | ||||||
|  |     ESP_LOGW(TAG, "Cannot notify GATT characteristic, not connected"); | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED); |     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED); | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (this->address_ != msg.address) { |   auto err = connection->notify_characteristic(msg.handle, msg.enable); | ||||||
|     ESP_LOGW(TAG, "Address mismatch for notify"); |   if (err != ESP_OK) { | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_WRONG_ADDRESS); |     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err); | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   auto *characteristic = this->get_characteristic(msg.handle); |  | ||||||
|  |  | ||||||
|   if (characteristic == nullptr) { |  | ||||||
|     ESP_LOGW(TAG, "Cannot notify GATT characteristic, not found."); |  | ||||||
|     api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_INVALID_HANDLE); |  | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   esp_err_t err; |  | ||||||
|   if (msg.enable) { |  | ||||||
|     err = esp_ble_gattc_register_for_notify(this->gattc_if_, this->remote_bda_, characteristic->handle); |  | ||||||
|     if (err != ESP_OK) { |  | ||||||
|       ESP_LOGW(TAG, "esp_ble_gattc_register_for_notify failed, err=%d", err); |  | ||||||
|       api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err); |  | ||||||
|     } |  | ||||||
|   } else { |  | ||||||
|     err = esp_ble_gattc_unregister_for_notify(this->gattc_if_, this->remote_bda_, characteristic->handle); |  | ||||||
|     if (err != ESP_OK) { |  | ||||||
|       ESP_LOGW(TAG, "esp_ble_gattc_unregister_for_notify failed, err=%d", err); |  | ||||||
|       api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -11,22 +11,26 @@ | |||||||
| #include "esphome/core/component.h" | #include "esphome/core/component.h" | ||||||
| #include "esphome/core/defines.h" | #include "esphome/core/defines.h" | ||||||
|  |  | ||||||
| #include <map> | #include "bluetooth_connection.h" | ||||||
|  |  | ||||||
| namespace esphome { | namespace esphome { | ||||||
| namespace bluetooth_proxy { | namespace bluetooth_proxy { | ||||||
|  |  | ||||||
|  | static const esp_err_t ESP_GATT_NOT_CONNECTED = -1; | ||||||
|  |  | ||||||
| using namespace esp32_ble_client; | using namespace esp32_ble_client; | ||||||
|  |  | ||||||
| class BluetoothProxy : public BLEClientBase { | class BluetoothProxy : public esp32_ble_tracker::ESPBTDeviceListener, public Component { | ||||||
|  public: |  public: | ||||||
|   BluetoothProxy(); |   BluetoothProxy(); | ||||||
|   bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override; |   bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override; | ||||||
|   void dump_config() override; |   void dump_config() override; | ||||||
|   void loop() override; |   void loop() override; | ||||||
|  |  | ||||||
|   void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, |   void register_connection(BluetoothConnection *connection) { | ||||||
|                            esp_ble_gattc_cb_param_t *param) override; |     this->connections_.push_back(connection); | ||||||
|  |     connection->proxy_ = this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   void bluetooth_device_request(const api::BluetoothDeviceRequest &msg); |   void bluetooth_device_request(const api::BluetoothDeviceRequest &msg); | ||||||
|   void bluetooth_gatt_read(const api::BluetoothGATTReadRequest &msg); |   void bluetooth_gatt_read(const api::BluetoothGATTReadRequest &msg); | ||||||
| @@ -36,8 +40,16 @@ class BluetoothProxy : public BLEClientBase { | |||||||
|   void bluetooth_gatt_send_services(const api::BluetoothGATTGetServicesRequest &msg); |   void bluetooth_gatt_send_services(const api::BluetoothGATTGetServicesRequest &msg); | ||||||
|   void bluetooth_gatt_notify(const api::BluetoothGATTNotifyRequest &msg); |   void bluetooth_gatt_notify(const api::BluetoothGATTNotifyRequest &msg); | ||||||
|  |  | ||||||
|   int get_bluetooth_connections_free() { return this->state_ == espbt::ClientState::IDLE ? 1 : 0; } |   int get_bluetooth_connections_free() { | ||||||
|   int get_bluetooth_connections_limit() { return 1; } |     int free = 0; | ||||||
|  |     for (auto *connection : this->connections_) { | ||||||
|  |       if (connection->address_ == 0) { | ||||||
|  |         free++; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return free; | ||||||
|  |   } | ||||||
|  |   int get_bluetooth_connections_limit() { return this->connections_.size(); } | ||||||
|  |  | ||||||
|   void set_active(bool active) { this->active_ = active; } |   void set_active(bool active) { this->active_ = active; } | ||||||
|   bool has_active() { return this->active_; } |   bool has_active() { return this->active_; } | ||||||
| @@ -45,8 +57,12 @@ class BluetoothProxy : public BLEClientBase { | |||||||
|  protected: |  protected: | ||||||
|   void send_api_packet_(const esp32_ble_tracker::ESPBTDevice &device); |   void send_api_packet_(const esp32_ble_tracker::ESPBTDevice &device); | ||||||
|  |  | ||||||
|  |   BluetoothConnection *get_connection_(uint64_t address, bool reserve); | ||||||
|  |  | ||||||
|   int16_t send_service_{-1}; |   int16_t send_service_{-1}; | ||||||
|   bool active_; |   bool active_; | ||||||
|  |  | ||||||
|  |   std::vector<BluetoothConnection *> connections_{}; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| extern BluetoothProxy *global_bluetooth_proxy;  // NOLINT(cppcoreguidelines-avoid-non-const-global-variables) | extern BluetoothProxy *global_bluetooth_proxy;  // NOLINT(cppcoreguidelines-avoid-non-const-global-variables) | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ | |||||||
| namespace esphome { | namespace esphome { | ||||||
| namespace esp32_ble_client { | namespace esp32_ble_client { | ||||||
|  |  | ||||||
| static const char *const TAG = "esp32_ble_client.characteristic"; | static const char *const TAG = "esp32_ble_client"; | ||||||
|  |  | ||||||
| BLECharacteristic::~BLECharacteristic() { | BLECharacteristic::~BLECharacteristic() { | ||||||
|   for (auto &desc : this->descriptors) |   for (auto &desc : this->descriptors) | ||||||
| @@ -29,7 +29,8 @@ void BLECharacteristic::parse_descriptors() { | |||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
|     if (status != ESP_GATT_OK) { |     if (status != ESP_GATT_OK) { | ||||||
|       ESP_LOGW(TAG, "esp_ble_gattc_get_all_descr error, status=%d", status); |       ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_get_all_descr error, status=%d", | ||||||
|  |                this->service->client->get_connection_index(), this->service->client->address_str().c_str(), status); | ||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
|     if (count == 0) { |     if (count == 0) { | ||||||
| @@ -41,7 +42,8 @@ void BLECharacteristic::parse_descriptors() { | |||||||
|     desc->handle = result.handle; |     desc->handle = result.handle; | ||||||
|     desc->characteristic = this; |     desc->characteristic = this; | ||||||
|     this->descriptors.push_back(desc); |     this->descriptors.push_back(desc); | ||||||
|     ESP_LOGV(TAG, "   descriptor %s, handle 0x%x", desc->uuid.to_string().c_str(), desc->handle); |     ESP_LOGV(TAG, "[%d] [%s]    descriptor %s, handle 0x%x", this->service->client->get_connection_index(), | ||||||
|  |              this->service->client->address_str().c_str(), desc->uuid.to_string().c_str(), desc->handle); | ||||||
|     offset++; |     offset++; | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @@ -69,7 +71,8 @@ esp_err_t BLECharacteristic::write_value(uint8_t *new_val, int16_t new_val_size, | |||||||
|   auto status = esp_ble_gattc_write_char(client->get_gattc_if(), client->get_conn_id(), this->handle, new_val_size, |   auto status = esp_ble_gattc_write_char(client->get_gattc_if(), client->get_conn_id(), this->handle, new_val_size, | ||||||
|                                          new_val, write_type, ESP_GATT_AUTH_REQ_NONE); |                                          new_val, write_type, ESP_GATT_AUTH_REQ_NONE); | ||||||
|   if (status) { |   if (status) { | ||||||
|     ESP_LOGW(TAG, "Error sending write value to BLE gattc server, status=%d", status); |     ESP_LOGW(TAG, "[%d] [%s] Error sending write value to BLE gattc server, status=%d", | ||||||
|  |              this->service->client->get_connection_index(), this->service->client->address_str().c_str(), status); | ||||||
|   } |   } | ||||||
|   return status; |   return status; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -11,6 +11,9 @@ namespace esp32_ble_client { | |||||||
| static const char *const TAG = "esp32_ble_client"; | static const char *const TAG = "esp32_ble_client"; | ||||||
|  |  | ||||||
| void BLEClientBase::setup() { | void BLEClientBase::setup() { | ||||||
|  |   static uint8_t connection_index = 0; | ||||||
|  |   this->connection_index_ = connection_index++; | ||||||
|  |  | ||||||
|   auto ret = esp_ble_gattc_app_register(this->app_id); |   auto ret = esp_ble_gattc_app_register(this->app_id); | ||||||
|   if (ret) { |   if (ret) { | ||||||
|     ESP_LOGE(TAG, "gattc app register failed. app_id=%d code=%d", this->app_id, ret); |     ESP_LOGE(TAG, "gattc app register failed. app_id=%d code=%d", this->app_id, ret); | ||||||
| @@ -33,7 +36,7 @@ bool BLEClientBase::parse_device(const espbt::ESPBTDevice &device) { | |||||||
|   if (this->state_ != espbt::ClientState::IDLE && this->state_ != espbt::ClientState::SEARCHING) |   if (this->state_ != espbt::ClientState::IDLE && this->state_ != espbt::ClientState::SEARCHING) | ||||||
|     return false; |     return false; | ||||||
|  |  | ||||||
|   ESP_LOGD(TAG, "Found device at MAC address [%s]", device.address_str().c_str()); |   ESP_LOGD(TAG, "[%d] [%s] Found device", this->connection_index_, this->address_str_.c_str()); | ||||||
|   this->set_state(espbt::ClientState::DISCOVERED); |   this->set_state(espbt::ClientState::DISCOVERED); | ||||||
|  |  | ||||||
|   auto addr = device.address_uint64(); |   auto addr = device.address_uint64(); | ||||||
| @@ -47,80 +50,88 @@ bool BLEClientBase::parse_device(const espbt::ESPBTDevice &device) { | |||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| std::string BLEClientBase::address_str() const { |  | ||||||
|   return str_snprintf("%02x:%02x:%02x:%02x:%02x:%02x", 17, (uint8_t)(this->address_ >> 40) & 0xff, |  | ||||||
|                       (uint8_t)(this->address_ >> 32) & 0xff, (uint8_t)(this->address_ >> 24) & 0xff, |  | ||||||
|                       (uint8_t)(this->address_ >> 16) & 0xff, (uint8_t)(this->address_ >> 8) & 0xff, |  | ||||||
|                       (uint8_t)(this->address_ >> 0) & 0xff); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void BLEClientBase::connect() { | void BLEClientBase::connect() { | ||||||
|   ESP_LOGI(TAG, "Attempting BLE connection to %s", this->address_str().c_str()); |   ESP_LOGI(TAG, "[%d] [%s] Attempting BLE connection", this->connection_index_, this->address_str_.c_str()); | ||||||
|   auto ret = esp_ble_gattc_open(this->gattc_if_, this->remote_bda_, this->remote_addr_type_, true); |   auto ret = esp_ble_gattc_open(this->gattc_if_, this->remote_bda_, this->remote_addr_type_, true); | ||||||
|   if (ret) { |   if (ret) { | ||||||
|     ESP_LOGW(TAG, "esp_ble_gattc_open error, address=%s status=%d", this->address_str().c_str(), ret); |     ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_open error, status=%d", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |              ret); | ||||||
|     this->set_state(espbt::ClientState::IDLE); |     this->set_state(espbt::ClientState::IDLE); | ||||||
|   } else { |   } else { | ||||||
|     this->set_state(espbt::ClientState::CONNECTING); |     this->set_state(espbt::ClientState::CONNECTING); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| void BLEClientBase::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t esp_gattc_if, | void BLEClientBase::disconnect() { | ||||||
|  |   ESP_LOGI(TAG, "[%d] [%s] Disconnecting.", this->connection_index_, this->address_str_.c_str()); | ||||||
|  |   auto err = esp_ble_gattc_close(this->gattc_if_, this->conn_id_); | ||||||
|  |   if (err != ESP_OK) { | ||||||
|  |     ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_close error, err=%d", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |              err); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (this->state_ == espbt::ClientState::SEARCHING) { | ||||||
|  |     this->set_address(0); | ||||||
|  |     this->set_state(espbt::ClientState::IDLE); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool BLEClientBase::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t esp_gattc_if, | ||||||
|                                         esp_ble_gattc_cb_param_t *param) { |                                         esp_ble_gattc_cb_param_t *param) { | ||||||
|   if (event == ESP_GATTC_REG_EVT && this->app_id != param->reg.app_id) |   if (event == ESP_GATTC_REG_EVT && this->app_id != param->reg.app_id) | ||||||
|     return; |     return false; | ||||||
|   if (event != ESP_GATTC_REG_EVT && esp_gattc_if != ESP_GATT_IF_NONE && esp_gattc_if != this->gattc_if_) |   if (event != ESP_GATTC_REG_EVT && esp_gattc_if != ESP_GATT_IF_NONE && esp_gattc_if != this->gattc_if_) | ||||||
|     return; |     return false; | ||||||
|  |  | ||||||
|  |   ESP_LOGV(TAG, "[%d] [%s] gattc_event_handler: event=%d gattc_if=%d", this->connection_index_, | ||||||
|  |            this->address_str_.c_str(), event, esp_gattc_if); | ||||||
|  |  | ||||||
|   switch (event) { |   switch (event) { | ||||||
|     case ESP_GATTC_REG_EVT: { |     case ESP_GATTC_REG_EVT: { | ||||||
|       if (param->reg.status == ESP_GATT_OK) { |       if (param->reg.status == ESP_GATT_OK) { | ||||||
|         ESP_LOGV(TAG, "gattc registered app id %d", this->app_id); |         ESP_LOGV(TAG, "[%d] [%s] gattc registered app id %d", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |                  this->app_id); | ||||||
|         this->gattc_if_ = esp_gattc_if; |         this->gattc_if_ = esp_gattc_if; | ||||||
|       } else { |       } else { | ||||||
|         ESP_LOGE(TAG, "gattc app registration failed id=%d code=%d", param->reg.app_id, param->reg.status); |         ESP_LOGE(TAG, "[%d] [%s] gattc app registration failed id=%d code=%d", this->connection_index_, | ||||||
|  |                  this->address_str_.c_str(), param->reg.app_id, param->reg.status); | ||||||
|       } |       } | ||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
|     case ESP_GATTC_OPEN_EVT: { |     case ESP_GATTC_OPEN_EVT: { | ||||||
|       ESP_LOGV(TAG, "[%s] ESP_GATTC_OPEN_EVT", this->address_str().c_str()); |       ESP_LOGV(TAG, "[%d] [%s] ESP_GATTC_OPEN_EVT", this->connection_index_, this->address_str_.c_str()); | ||||||
|       this->conn_id_ = param->open.conn_id; |       this->conn_id_ = param->open.conn_id; | ||||||
|       if (param->open.status != ESP_GATT_OK && param->open.status != ESP_GATT_ALREADY_OPEN) { |       if (param->open.status != ESP_GATT_OK && param->open.status != ESP_GATT_ALREADY_OPEN) { | ||||||
|         ESP_LOGW(TAG, "connect to %s failed, status=%d", this->address_str().c_str(), param->open.status); |         ESP_LOGW(TAG, "[%d] [%s] Connection failed, status=%d", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |                  param->open.status); | ||||||
|         this->set_state(espbt::ClientState::IDLE); |         this->set_state(espbt::ClientState::IDLE); | ||||||
|         break; |         break; | ||||||
|       } |       } | ||||||
|       break; |       auto ret = esp_ble_gattc_send_mtu_req(this->gattc_if_, param->open.conn_id); | ||||||
|     } |  | ||||||
|     case ESP_GATTC_CONNECT_EVT: { |  | ||||||
|       ESP_LOGV(TAG, "[%s] ESP_GATTC_CONNECT_EVT", this->address_str().c_str()); |  | ||||||
|       if (this->conn_id_ != param->connect.conn_id) { |  | ||||||
|         ESP_LOGD(TAG, "[%s] Unexpected conn_id in CONNECT_EVT: param conn=%d, open conn=%d", |  | ||||||
|                  this->address_str().c_str(), param->connect.conn_id, this->conn_id_); |  | ||||||
|       } |  | ||||||
|       auto ret = esp_ble_gattc_send_mtu_req(this->gattc_if_, param->connect.conn_id); |  | ||||||
|       if (ret) { |       if (ret) { | ||||||
|         ESP_LOGW(TAG, "esp_ble_gattc_send_mtu_req failed, status=%x", ret); |         ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_send_mtu_req failed, status=%x", this->connection_index_, | ||||||
|  |                  this->address_str_.c_str(), ret); | ||||||
|       } |       } | ||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
|     case ESP_GATTC_CFG_MTU_EVT: { |     case ESP_GATTC_CFG_MTU_EVT: { | ||||||
|       if (param->cfg_mtu.status != ESP_GATT_OK) { |       if (param->cfg_mtu.status != ESP_GATT_OK) { | ||||||
|         ESP_LOGW(TAG, "cfg_mtu to %s failed, mtu %d, status %d", this->address_str().c_str(), param->cfg_mtu.mtu, |         ESP_LOGW(TAG, "[%d] [%s] cfg_mtu failed, mtu %d, status %d", this->connection_index_, | ||||||
|                  param->cfg_mtu.status); |                  this->address_str_.c_str(), param->cfg_mtu.mtu, param->cfg_mtu.status); | ||||||
|         this->set_state(espbt::ClientState::IDLE); |         this->set_state(espbt::ClientState::IDLE); | ||||||
|         break; |         break; | ||||||
|       } |       } | ||||||
|       ESP_LOGV(TAG, "cfg_mtu status %d, mtu %d", param->cfg_mtu.status, param->cfg_mtu.mtu); |       ESP_LOGV(TAG, "[%d] [%s] cfg_mtu status %d, mtu %d", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |                param->cfg_mtu.status, param->cfg_mtu.mtu); | ||||||
|       this->mtu_ = param->cfg_mtu.mtu; |       this->mtu_ = param->cfg_mtu.mtu; | ||||||
|       esp_ble_gattc_search_service(esp_gattc_if, param->cfg_mtu.conn_id, nullptr); |       esp_ble_gattc_search_service(esp_gattc_if, param->cfg_mtu.conn_id, nullptr); | ||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
|     case ESP_GATTC_DISCONNECT_EVT: { |     case ESP_GATTC_DISCONNECT_EVT: { | ||||||
|       if (memcmp(param->disconnect.remote_bda, this->remote_bda_, 6) != 0) { |       if (memcmp(param->disconnect.remote_bda, this->remote_bda_, 6) != 0) | ||||||
|         return; |         return false; | ||||||
|       } |       ESP_LOGV(TAG, "[%d] [%s] ESP_GATTC_DISCONNECT_EVT, reason %d", this->connection_index_, | ||||||
|       ESP_LOGV(TAG, "[%s] ESP_GATTC_DISCONNECT_EVT, reason %d", this->address_str().c_str(), param->disconnect.reason); |                this->address_str_.c_str(), param->disconnect.reason); | ||||||
|       for (auto &svc : this->services_) |       for (auto &svc : this->services_) | ||||||
|         delete svc;  // NOLINT(cppcoreguidelines-owning-memory) |         delete svc;  // NOLINT(cppcoreguidelines-owning-memory) | ||||||
|       this->services_.clear(); |       this->services_.clear(); | ||||||
| @@ -137,10 +148,12 @@ void BLEClientBase::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_ | |||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
|     case ESP_GATTC_SEARCH_CMPL_EVT: { |     case ESP_GATTC_SEARCH_CMPL_EVT: { | ||||||
|       ESP_LOGV(TAG, "[%s] ESP_GATTC_SEARCH_CMPL_EVT", this->address_str().c_str()); |       ESP_LOGV(TAG, "[%d] [%s] ESP_GATTC_SEARCH_CMPL_EVT", this->connection_index_, this->address_str_.c_str()); | ||||||
|       for (auto &svc : this->services_) { |       for (auto &svc : this->services_) { | ||||||
|         ESP_LOGI(TAG, "Service UUID: %s", svc->uuid.to_string().c_str()); |         ESP_LOGI(TAG, "[%d] [%s] Service UUID: %s", this->connection_index_, this->address_str_.c_str(), | ||||||
|         ESP_LOGI(TAG, "  start_handle: 0x%x  end_handle: 0x%x", svc->start_handle, svc->end_handle); |                  svc->uuid.to_string().c_str()); | ||||||
|  |         ESP_LOGI(TAG, "[%d] [%s]  start_handle: 0x%x  end_handle: 0x%x", this->connection_index_, | ||||||
|  |                  this->address_str_.c_str(), svc->start_handle, svc->end_handle); | ||||||
|         svc->parse_characteristics(); |         svc->parse_characteristics(); | ||||||
|       } |       } | ||||||
|       this->set_state(espbt::ClientState::CONNECTED); |       this->set_state(espbt::ClientState::CONNECTED); | ||||||
| @@ -149,14 +162,10 @@ void BLEClientBase::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_ | |||||||
|     } |     } | ||||||
|     case ESP_GATTC_REG_FOR_NOTIFY_EVT: { |     case ESP_GATTC_REG_FOR_NOTIFY_EVT: { | ||||||
|       auto *descr = this->get_config_descriptor(param->reg_for_notify.handle); |       auto *descr = this->get_config_descriptor(param->reg_for_notify.handle); | ||||||
|       if (descr == nullptr) { |  | ||||||
|         ESP_LOGW(TAG, "No descriptor found for notify of handle 0x%x", param->reg_for_notify.handle); |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       if (descr->uuid.get_uuid().len != ESP_UUID_LEN_16 || |       if (descr->uuid.get_uuid().len != ESP_UUID_LEN_16 || | ||||||
|           descr->uuid.get_uuid().uuid.uuid16 != ESP_GATT_UUID_CHAR_CLIENT_CONFIG) { |           descr->uuid.get_uuid().uuid.uuid16 != ESP_GATT_UUID_CHAR_CLIENT_CONFIG) { | ||||||
|         ESP_LOGW(TAG, "Handle 0x%x (uuid %s) is not a client config char uuid", param->reg_for_notify.handle, |         ESP_LOGW(TAG, "[%d] [%s] Handle 0x%x (uuid %s) is not a client config char uuid", this->connection_index_, | ||||||
|                  descr->uuid.to_string().c_str()); |                  this->address_str_.c_str(), param->reg_for_notify.handle, descr->uuid.to_string().c_str()); | ||||||
|         break; |         break; | ||||||
|       } |       } | ||||||
|       uint16_t notify_en = 1; |       uint16_t notify_en = 1; | ||||||
| @@ -164,7 +173,8 @@ void BLEClientBase::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_ | |||||||
|           esp_ble_gattc_write_char_descr(this->gattc_if_, this->conn_id_, descr->handle, sizeof(notify_en), |           esp_ble_gattc_write_char_descr(this->gattc_if_, this->conn_id_, descr->handle, sizeof(notify_en), | ||||||
|                                          (uint8_t *) ¬ify_en, ESP_GATT_WRITE_TYPE_RSP, ESP_GATT_AUTH_REQ_NONE); |                                          (uint8_t *) ¬ify_en, ESP_GATT_WRITE_TYPE_RSP, ESP_GATT_AUTH_REQ_NONE); | ||||||
|       if (status) { |       if (status) { | ||||||
|         ESP_LOGW(TAG, "esp_ble_gattc_write_char_descr error, status=%d", status); |         ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_write_char_descr error, status=%d", this->connection_index_, | ||||||
|  |                  this->address_str_.c_str(), status); | ||||||
|       } |       } | ||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
| @@ -172,24 +182,28 @@ void BLEClientBase::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_ | |||||||
|     default: |     default: | ||||||
|       break; |       break; | ||||||
|   } |   } | ||||||
|  |   return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| void BLEClientBase::gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) { | void BLEClientBase::gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) { | ||||||
|   switch (event) { |   switch (event) { | ||||||
|     // This event is sent by the server when it requests security |     // This event is sent by the server when it requests security | ||||||
|     case ESP_GAP_BLE_SEC_REQ_EVT: |     case ESP_GAP_BLE_SEC_REQ_EVT: | ||||||
|       ESP_LOGV(TAG, "ESP_GAP_BLE_SEC_REQ_EVT %x", event); |       ESP_LOGV(TAG, "[%d] [%s] ESP_GAP_BLE_SEC_REQ_EVT %x", this->connection_index_, this->address_str_.c_str(), event); | ||||||
|       esp_ble_gap_security_rsp(param->ble_security.ble_req.bd_addr, true); |       esp_ble_gap_security_rsp(param->ble_security.ble_req.bd_addr, true); | ||||||
|       break; |       break; | ||||||
|     // This event is sent once authentication has completed |     // This event is sent once authentication has completed | ||||||
|     case ESP_GAP_BLE_AUTH_CMPL_EVT: |     case ESP_GAP_BLE_AUTH_CMPL_EVT: | ||||||
|       esp_bd_addr_t bd_addr; |       esp_bd_addr_t bd_addr; | ||||||
|       memcpy(bd_addr, param->ble_security.auth_cmpl.bd_addr, sizeof(esp_bd_addr_t)); |       memcpy(bd_addr, param->ble_security.auth_cmpl.bd_addr, sizeof(esp_bd_addr_t)); | ||||||
|       ESP_LOGI(TAG, "auth complete. remote BD_ADDR: %s", format_hex(bd_addr, 6).c_str()); |       ESP_LOGI(TAG, "[%d] [%s] auth complete. remote BD_ADDR: %s", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |                format_hex(bd_addr, 6).c_str()); | ||||||
|       if (!param->ble_security.auth_cmpl.success) { |       if (!param->ble_security.auth_cmpl.success) { | ||||||
|         ESP_LOGE(TAG, "auth fail reason = 0x%x", param->ble_security.auth_cmpl.fail_reason); |         ESP_LOGE(TAG, "[%d] [%s] auth fail reason = 0x%x", this->connection_index_, this->address_str_.c_str(), | ||||||
|  |                  param->ble_security.auth_cmpl.fail_reason); | ||||||
|       } else { |       } else { | ||||||
|         ESP_LOGV(TAG, "auth success. address type = %d auth mode = %d", param->ble_security.auth_cmpl.addr_type, |         ESP_LOGV(TAG, "[%d] [%s] auth success. address type = %d auth mode = %d", this->connection_index_, | ||||||
|  |                  this->address_str_.c_str(), param->ble_security.auth_cmpl.addr_type, | ||||||
|                  param->ble_security.auth_cmpl.auth_mode); |                  param->ble_security.auth_cmpl.auth_mode); | ||||||
|       } |       } | ||||||
|       break; |       break; | ||||||
| @@ -245,7 +259,8 @@ float BLEClientBase::parse_char_value(uint8_t *value, uint16_t length) { | |||||||
|                         (int32_t)(value[4])); |                         (int32_t)(value[4])); | ||||||
|       } |       } | ||||||
|   } |   } | ||||||
|   ESP_LOGW(TAG, "Cannot parse characteristic value of type 0x%x length %d", value[0], length); |   ESP_LOGW(TAG, "[%d] [%s] Cannot parse characteristic value of type 0x%x length %d", this->connection_index_, | ||||||
|  |            this->address_str_.c_str(), value[0], length); | ||||||
|   return NAN; |   return NAN; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -28,13 +28,27 @@ class BLEClientBase : public espbt::ESPBTClient, public Component { | |||||||
|  |  | ||||||
|   bool parse_device(const espbt::ESPBTDevice &device) override; |   bool parse_device(const espbt::ESPBTDevice &device) override; | ||||||
|   void on_scan_end() override {} |   void on_scan_end() override {} | ||||||
|   void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, |   bool gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, | ||||||
|                            esp_ble_gattc_cb_param_t *param) override; |                            esp_ble_gattc_cb_param_t *param) override; | ||||||
|   void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) override; |   void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) override; | ||||||
|   void connect() override; |   void connect() override; | ||||||
|  |   void disconnect(); | ||||||
|  |  | ||||||
|   void set_address(uint64_t address) { this->address_ = address; } |   bool connected() { return this->state_ == espbt::ClientState::ESTABLISHED; } | ||||||
|   std::string address_str() const; |  | ||||||
|  |   void set_address(uint64_t address) { | ||||||
|  |     this->address_ = address; | ||||||
|  |     if (address == 0) { | ||||||
|  |       memset(this->remote_bda_, 0, sizeof(this->remote_bda_)); | ||||||
|  |       this->address_str_ = ""; | ||||||
|  |     } else { | ||||||
|  |       this->address_str_ = str_snprintf("%02x:%02x:%02x:%02x:%02x:%02x", 17, (uint8_t)(this->address_ >> 40) & 0xff, | ||||||
|  |                                         (uint8_t)(this->address_ >> 32) & 0xff, (uint8_t)(this->address_ >> 24) & 0xff, | ||||||
|  |                                         (uint8_t)(this->address_ >> 16) & 0xff, (uint8_t)(this->address_ >> 8) & 0xff, | ||||||
|  |                                         (uint8_t)(this->address_ >> 0) & 0xff); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   std::string address_str() const { return this->address_str_; } | ||||||
|  |  | ||||||
|   BLEService *get_service(espbt::ESPBTUUID uuid); |   BLEService *get_service(espbt::ESPBTUUID uuid); | ||||||
|   BLEService *get_service(uint16_t uuid); |   BLEService *get_service(uint16_t uuid); | ||||||
| @@ -55,12 +69,16 @@ class BLEClientBase : public espbt::ESPBTClient, public Component { | |||||||
|   uint16_t get_conn_id() const { return this->conn_id_; } |   uint16_t get_conn_id() const { return this->conn_id_; } | ||||||
|   uint64_t get_address() const { return this->address_; } |   uint64_t get_address() const { return this->address_; } | ||||||
|  |  | ||||||
|  |   uint8_t get_connection_index() const { return this->connection_index_; } | ||||||
|  |  | ||||||
|  protected: |  protected: | ||||||
|   int gattc_if_; |   int gattc_if_; | ||||||
|   esp_bd_addr_t remote_bda_; |   esp_bd_addr_t remote_bda_; | ||||||
|   esp_ble_addr_type_t remote_addr_type_; |   esp_ble_addr_type_t remote_addr_type_; | ||||||
|   uint16_t conn_id_; |   uint16_t conn_id_{0xFFFF}; | ||||||
|   uint64_t address_; |   uint64_t address_{0}; | ||||||
|  |   std::string address_str_{}; | ||||||
|  |   uint8_t connection_index_; | ||||||
|   uint16_t mtu_{23}; |   uint16_t mtu_{23}; | ||||||
|  |  | ||||||
|   std::vector<BLEService *> services_; |   std::vector<BLEService *> services_; | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
| namespace esphome { | namespace esphome { | ||||||
| namespace esp32_ble_client { | namespace esp32_ble_client { | ||||||
|  |  | ||||||
| static const char *const TAG = "esp32_ble_client.service"; | static const char *const TAG = "esp32_ble_client"; | ||||||
|  |  | ||||||
| BLECharacteristic *BLEService::get_characteristic(espbt::ESPBTUUID uuid) { | BLECharacteristic *BLEService::get_characteristic(espbt::ESPBTUUID uuid) { | ||||||
|   for (auto &chr : this->characteristics) { |   for (auto &chr : this->characteristics) { | ||||||
| @@ -40,7 +40,8 @@ void BLEService::parse_characteristics() { | |||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
|     if (status != ESP_GATT_OK) { |     if (status != ESP_GATT_OK) { | ||||||
|       ESP_LOGW(TAG, "esp_ble_gattc_get_all_char error, status=%d", status); |       ESP_LOGW(TAG, "[%d] [%s] esp_ble_gattc_get_all_char error, status=%d", this->client->get_connection_index(), | ||||||
|  |                this->client->address_str().c_str(), status); | ||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
|     if (count == 0) { |     if (count == 0) { | ||||||
| @@ -53,8 +54,9 @@ void BLEService::parse_characteristics() { | |||||||
|     characteristic->handle = result.char_handle; |     characteristic->handle = result.char_handle; | ||||||
|     characteristic->service = this; |     characteristic->service = this; | ||||||
|     this->characteristics.push_back(characteristic); |     this->characteristics.push_back(characteristic); | ||||||
|     ESP_LOGI(TAG, " characteristic %s, handle 0x%x, properties 0x%x", characteristic->uuid.to_string().c_str(), |     ESP_LOGI(TAG, "[%d] [%s]  characteristic %s, handle 0x%x, properties 0x%x", this->client->get_connection_index(), | ||||||
|              characteristic->handle, characteristic->properties); |              this->client->address_str().c_str(), characteristic->uuid.to_string().c_str(), characteristic->handle, | ||||||
|  |              characteristic->properties); | ||||||
|     characteristic->parse_descriptors(); |     characteristic->parse_descriptors(); | ||||||
|     offset++; |     offset++; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -159,7 +159,7 @@ enum class ClientState { | |||||||
|  |  | ||||||
| class ESPBTClient : public ESPBTDeviceListener { | class ESPBTClient : public ESPBTDeviceListener { | ||||||
|  public: |  public: | ||||||
|   virtual void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, |   virtual bool gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, | ||||||
|                                    esp_ble_gattc_cb_param_t *param) = 0; |                                    esp_ble_gattc_cb_param_t *param) = 0; | ||||||
|   virtual void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) = 0; |   virtual void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) = 0; | ||||||
|   virtual void connect() = 0; |   virtual void connect() = 0; | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user