From a47a70957b12afa42c7e813cc7ec81f609892dc4 Mon Sep 17 00:00:00 2001 From: Emanuel Posescu Date: Sun, 24 Jan 2021 21:23:40 +0200 Subject: [PATCH 1/2] Cleanup in case of registration and connect failure. Cleanup before calling disconnect callback for safe delete. Reject other events during registration. Adresses #4047, #4055 --- libraries/BLE/src/BLEClient.cpp | 33 +++++++++++++++++++++++++++------ 1 file changed, 27 insertions(+), 6 deletions(-) diff --git a/libraries/BLE/src/BLEClient.cpp b/libraries/BLE/src/BLEClient.cpp index 9d947710e94..4036e68f44b 100644 --- a/libraries/BLE/src/BLEClient.cpp +++ b/libraries/BLE/src/BLEClient.cpp @@ -109,7 +109,16 @@ bool BLEClient::connect(BLEAddress address, esp_ble_addr_type_t type) { return false; } - m_semaphoreRegEvt.wait("connect"); + uint32_t rc = m_semaphoreRegEvt.wait("connect"); + + if (rc != ESP_GATT_OK) { + // fixes ESP_GATT_NO_RESOURCES error mostly + log_e("esp_ble_gattc_app_register_error: rc=%d", rc); + BLEDevice::removePeerDevice(m_appId, true); + // not sure if this is needed here + // esp_ble_gattc_app_unregister(m_gattc_if); + return false; + } m_peerAddress = address; @@ -127,7 +136,12 @@ bool BLEClient::connect(BLEAddress address, esp_ble_addr_type_t type) { return false; } - uint32_t rc = m_semaphoreOpenEvt.wait("connect"); // Wait for the connection to complete. + rc = m_semaphoreOpenEvt.wait("connect"); // Wait for the connection to complete. + // check the status of the connection and cleanup in case of failure + if (rc != ESP_GATT_OK) { + BLEDevice::removePeerDevice(m_appId, true); + esp_ble_gattc_app_unregister(m_gattc_if); + } log_v("<< connect(), rc=%d", rc==ESP_GATT_OK); return rc == ESP_GATT_OK; } // connect @@ -159,6 +173,11 @@ void BLEClient::gattClientEventHandler( log_d("gattClientEventHandler [esp_gatt_if: %d] ... %s", gattc_if, BLEUtils::gattClientEventTypeToString(event).c_str()); + // it is possible to receive events from other connections while waiting for registration + if (m_gattc_if == ESP_GATT_IF_NONE && event != ESP_GATTC_REG_EVT) { + return; + } + // Execute handler code based on the type of event received. switch(event) { @@ -183,15 +202,16 @@ void BLEClient::gattClientEventHandler( if (evtParam->disconnect.conn_id != getConnId()) break; // If we receive a disconnect event, set the class flag that indicates that we are // no longer connected. - if (m_isConnected && m_pClientCallbacks != nullptr) { - m_pClientCallbacks->onDisconnect(this); - } + bool m_wasConnected = m_isConnected; m_isConnected = false; esp_ble_gattc_app_unregister(m_gattc_if); m_semaphoreOpenEvt.give(ESP_GATT_IF_NONE); m_semaphoreRssiCmplEvt.give(); m_semaphoreSearchCmplEvt.give(1); BLEDevice::removePeerDevice(m_appId, true); + if (m_wasConnected && m_pClientCallbacks != nullptr) { + m_pClientCallbacks->onDisconnect(this); + } break; } // ESP_GATTC_DISCONNECT_EVT @@ -227,7 +247,8 @@ void BLEClient::gattClientEventHandler( // case ESP_GATTC_REG_EVT: { m_gattc_if = gattc_if; - m_semaphoreRegEvt.give(); + // pass on the registration status result, in case of failure + m_semaphoreRegEvt.give(evtParam->reg.status); break; } // ESP_GATTC_REG_EVT From 522df520137bdd5fcdeb96bece9d35de5079ab8f Mon Sep 17 00:00:00 2001 From: Emanuel Posescu Date: Mon, 25 Jan 2021 09:57:50 +0200 Subject: [PATCH 2/2] Clear if after unregister #4047 --- libraries/BLE/src/BLEClient.cpp | 3 +++ 1 file changed, 3 insertions(+) diff --git a/libraries/BLE/src/BLEClient.cpp b/libraries/BLE/src/BLEClient.cpp index 4036e68f44b..13031f30450 100644 --- a/libraries/BLE/src/BLEClient.cpp +++ b/libraries/BLE/src/BLEClient.cpp @@ -117,6 +117,7 @@ bool BLEClient::connect(BLEAddress address, esp_ble_addr_type_t type) { BLEDevice::removePeerDevice(m_appId, true); // not sure if this is needed here // esp_ble_gattc_app_unregister(m_gattc_if); + // m_gattc_if = ESP_GATT_IF_NONE; return false; } @@ -141,6 +142,7 @@ bool BLEClient::connect(BLEAddress address, esp_ble_addr_type_t type) { if (rc != ESP_GATT_OK) { BLEDevice::removePeerDevice(m_appId, true); esp_ble_gattc_app_unregister(m_gattc_if); + m_gattc_if = ESP_GATT_IF_NONE; } log_v("<< connect(), rc=%d", rc==ESP_GATT_OK); return rc == ESP_GATT_OK; @@ -205,6 +207,7 @@ void BLEClient::gattClientEventHandler( bool m_wasConnected = m_isConnected; m_isConnected = false; esp_ble_gattc_app_unregister(m_gattc_if); + m_gattc_if = ESP_GATT_IF_NONE; m_semaphoreOpenEvt.give(ESP_GATT_IF_NONE); m_semaphoreRssiCmplEvt.give(); m_semaphoreSearchCmplEvt.give(1);