mirror of
https://github.com/h2zero/esp-nimble-cpp.git
synced 2024-11-21 20:50:55 +01:00
Refactor client connection establishment and client deletion.
* Removes the connection established flag as it should not be necessary. * Deleting the client instance from the `onDisconnect` callback is now supported. * `NimBLEDevice::deleteClient` no longer blocks tasks * Adds a new `Config` struct to `NimBLEClient` to efficiently set single bit config settings. * Adds `NimBLEClient::setSelfDelete` that takes the bool parameters `deleteOnDisconnect` and `deleteOnConnectFail` This will configure the client to delete itself when disconnected or the connection attempt fails. * Adds `NimBLEClient::setConfig` and `NimBLEClient::getConfig` which takes or returns a `NimBLEClient::Config` object respectively. * Reword `BLE_HS_EAPP` error string to be more accurate.
This commit is contained in:
parent
a6e75b3537
commit
fbbcfadc0c
7 changed files with 146 additions and 100 deletions
|
@ -63,10 +63,7 @@ NimBLEClient::NimBLEClient(const NimBLEAddress& peerAddress)
|
||||||
m_pClientCallbacks{&defaultCallbacks},
|
m_pClientCallbacks{&defaultCallbacks},
|
||||||
m_connHandle{BLE_HS_CONN_HANDLE_NONE},
|
m_connHandle{BLE_HS_CONN_HANDLE_NONE},
|
||||||
m_terminateFailCount{0},
|
m_terminateFailCount{0},
|
||||||
m_deleteCallbacks{false},
|
m_config{},
|
||||||
m_connEstablished{false},
|
|
||||||
m_asyncConnect{false},
|
|
||||||
m_exchangeMTU{true},
|
|
||||||
# if CONFIG_BT_NIMBLE_EXT_ADV
|
# if CONFIG_BT_NIMBLE_EXT_ADV
|
||||||
m_phyMask{BLE_GAP_LE_PHY_1M_MASK | BLE_GAP_LE_PHY_2M_MASK | BLE_GAP_LE_PHY_CODED_MASK},
|
m_phyMask{BLE_GAP_LE_PHY_1M_MASK | BLE_GAP_LE_PHY_2M_MASK | BLE_GAP_LE_PHY_CODED_MASK},
|
||||||
# endif
|
# endif
|
||||||
|
@ -89,7 +86,7 @@ NimBLEClient::~NimBLEClient() {
|
||||||
// Before we are finished with the client, we must release resources.
|
// Before we are finished with the client, we must release resources.
|
||||||
deleteServices();
|
deleteServices();
|
||||||
|
|
||||||
if (m_deleteCallbacks) {
|
if (m_config.deleteCallbacks) {
|
||||||
delete m_pClientCallbacks;
|
delete m_pClientCallbacks;
|
||||||
}
|
}
|
||||||
} // ~NimBLEClient
|
} // ~NimBLEClient
|
||||||
|
@ -122,7 +119,7 @@ size_t NimBLEClient::deleteService(const NimBLEUUID& uuid) {
|
||||||
}
|
}
|
||||||
|
|
||||||
return m_svcVec.size();
|
return m_svcVec.size();
|
||||||
} // deleteServices
|
} // deleteService
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Connect to the BLE Server using the address of the last connected device, or the address\n
|
* @brief Connect to the BLE Server using the address of the last connected device, or the address\n
|
||||||
|
@ -174,7 +171,7 @@ bool NimBLEClient::connect(const NimBLEAddress& address, bool deleteAttributes,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isConnected() || m_connEstablished) {
|
if (isConnected()) {
|
||||||
NIMBLE_LOGE(LOG_TAG, "Client already connected");
|
NIMBLE_LOGE(LOG_TAG, "Client already connected");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -202,8 +199,8 @@ bool NimBLEClient::connect(const NimBLEAddress& address, bool deleteAttributes,
|
||||||
}
|
}
|
||||||
|
|
||||||
int rc = 0;
|
int rc = 0;
|
||||||
m_asyncConnect = asyncConnect;
|
m_config.asyncConnect = asyncConnect;
|
||||||
m_exchangeMTU = exchangeMTU;
|
m_config.exchangeMTU = exchangeMTU;
|
||||||
|
|
||||||
// Set the connection in progress flag to prevent a scan from starting while connecting.
|
// Set the connection in progress flag to prevent a scan from starting while connecting.
|
||||||
NimBLEDevice::setConnectionInProgress(true);
|
NimBLEDevice::setConnectionInProgress(true);
|
||||||
|
@ -265,7 +262,7 @@ bool NimBLEClient::connect(const NimBLEAddress& address, bool deleteAttributes,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (m_asyncConnect) {
|
if (m_config.asyncConnect) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -289,19 +286,14 @@ bool NimBLEClient::connect(const NimBLEAddress& address, bool deleteAttributes,
|
||||||
rc = taskData.m_flags;
|
rc = taskData.m_flags;
|
||||||
if (rc != 0) {
|
if (rc != 0) {
|
||||||
NIMBLE_LOGE(LOG_TAG, "Connection failed; status=%d %s", rc, NimBLEUtils::returnCodeToString(rc));
|
NIMBLE_LOGE(LOG_TAG, "Connection failed; status=%d %s", rc, NimBLEUtils::returnCodeToString(rc));
|
||||||
// If the failure was not a result of a disconnection, make sure we disconnect now to avoid dangling connections
|
|
||||||
if (isConnected()) {
|
|
||||||
disconnect();
|
|
||||||
}
|
|
||||||
m_lastErr = rc;
|
m_lastErr = rc;
|
||||||
|
if (m_config.deleteOnConnectFail) {
|
||||||
|
NimBLEDevice::deleteClient(this);
|
||||||
|
}
|
||||||
return false;
|
return false;
|
||||||
} else {
|
|
||||||
NIMBLE_LOGI(LOG_TAG, "Connection established");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
m_connEstablished = true;
|
|
||||||
m_pClientCallbacks->onConnect(this);
|
m_pClientCallbacks->onConnect(this);
|
||||||
|
|
||||||
NIMBLE_LOGD(LOG_TAG, "<< connect()");
|
NIMBLE_LOGD(LOG_TAG, "<< connect()");
|
||||||
// Check if still connected before returning
|
// Check if still connected before returning
|
||||||
return isConnected();
|
return isConnected();
|
||||||
|
@ -357,7 +349,7 @@ bool NimBLEClient::disconnect(uint8_t reason) {
|
||||||
* @brief Cancel an ongoing connection attempt.
|
* @brief Cancel an ongoing connection attempt.
|
||||||
* @return True if the command was successfully sent.
|
* @return True if the command was successfully sent.
|
||||||
*/
|
*/
|
||||||
bool NimBLEClient::cancelConnect() {
|
bool NimBLEClient::cancelConnect() const {
|
||||||
int rc = ble_gap_conn_cancel();
|
int rc = ble_gap_conn_cancel();
|
||||||
if (rc != 0 && rc != BLE_HS_EALREADY) {
|
if (rc != 0 && rc != BLE_HS_EALREADY) {
|
||||||
NIMBLE_LOGE(LOG_TAG, "ble_gap_conn_cancel failed: rc=%d %s", rc, NimBLEUtils::returnCodeToString(rc));
|
NIMBLE_LOGE(LOG_TAG, "ble_gap_conn_cancel failed: rc=%d %s", rc, NimBLEUtils::returnCodeToString(rc));
|
||||||
|
@ -368,6 +360,32 @@ bool NimBLEClient::cancelConnect() {
|
||||||
return true;
|
return true;
|
||||||
} // cancelConnect
|
} // cancelConnect
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Set or unset a flag to delete this client when disconnected or connection failed.
|
||||||
|
* @param [in] deleteOnDisconnect Set the client to self delete when disconnected.
|
||||||
|
* @param [in] deleteOnConnectFail Set the client to self delete when a connection attempt fails.
|
||||||
|
*/
|
||||||
|
void NimBLEClient::setSelfDelete(bool deleteOnDisconnect, bool deleteOnConnectFail) {
|
||||||
|
m_config.deleteOnDisconnect = deleteOnDisconnect;
|
||||||
|
m_config.deleteOnConnectFail = deleteOnConnectFail;
|
||||||
|
} // setSelfDelete
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Get a copy of the clients configuration.
|
||||||
|
* @return A copy of the clients configuration.
|
||||||
|
*/
|
||||||
|
NimBLEClient::Config NimBLEClient::getConfig() const {
|
||||||
|
return m_config;
|
||||||
|
} // getConfig
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Set the client configuration options.
|
||||||
|
* @param [in] config The config options instance to set the client configuration to.
|
||||||
|
*/
|
||||||
|
void NimBLEClient::setConfig(NimBLEClient::Config config) {
|
||||||
|
m_config = config;
|
||||||
|
} // setConfig
|
||||||
|
|
||||||
# if CONFIG_BT_NIMBLE_EXT_ADV
|
# if CONFIG_BT_NIMBLE_EXT_ADV
|
||||||
/**
|
/**
|
||||||
* @brief Set the PHY types to use when connecting to a server.
|
* @brief Set the PHY types to use when connecting to a server.
|
||||||
|
@ -493,7 +511,6 @@ uint16_t NimBLEClient::getConnHandle() const {
|
||||||
*/
|
*/
|
||||||
void NimBLEClient::clearConnection() {
|
void NimBLEClient::clearConnection() {
|
||||||
m_connHandle = BLE_HS_CONN_HANDLE_NONE;
|
m_connHandle = BLE_HS_CONN_HANDLE_NONE;
|
||||||
m_connEstablished = false;
|
|
||||||
m_peerAddress = NimBLEAddress{};
|
m_peerAddress = NimBLEAddress{};
|
||||||
} // clearConnection
|
} // clearConnection
|
||||||
|
|
||||||
|
@ -509,15 +526,13 @@ void NimBLEClient::clearConnection() {
|
||||||
* This enables the GATT Server to read the attributes of the client connected to it.
|
* This enables the GATT Server to read the attributes of the client connected to it.
|
||||||
*/
|
*/
|
||||||
bool NimBLEClient::setConnection(const NimBLEConnInfo& connInfo) {
|
bool NimBLEClient::setConnection(const NimBLEConnInfo& connInfo) {
|
||||||
if (isConnected() || m_connEstablished) {
|
if (isConnected()) {
|
||||||
NIMBLE_LOGE(LOG_TAG, "Already connected");
|
NIMBLE_LOGE(LOG_TAG, "Already connected");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
m_peerAddress = connInfo.getAddress();
|
m_peerAddress = connInfo.getAddress();
|
||||||
m_connHandle = connInfo.getConnHandle();
|
m_connHandle = connInfo.getConnHandle();
|
||||||
m_connEstablished = true;
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
} // setConnection
|
} // setConnection
|
||||||
|
|
||||||
|
@ -763,6 +778,12 @@ int NimBLEClient::serviceDiscoveredCB(uint16_t connHandle,
|
||||||
NimBLETaskData* pTaskData = (NimBLETaskData*)arg;
|
NimBLETaskData* pTaskData = (NimBLETaskData*)arg;
|
||||||
NimBLEClient* pClient = (NimBLEClient*)pTaskData->m_pInstance;
|
NimBLEClient* pClient = (NimBLEClient*)pTaskData->m_pInstance;
|
||||||
|
|
||||||
|
if (error->status == BLE_HS_ENOTCONN) {
|
||||||
|
NIMBLE_LOGE(LOG_TAG, "<< Service Discovered; Not connected");
|
||||||
|
NimBLEUtils::taskRelease(*pTaskData, error->status);
|
||||||
|
return error->status;
|
||||||
|
}
|
||||||
|
|
||||||
// Make sure the service discovery is for this device
|
// Make sure the service discovery is for this device
|
||||||
if (pClient->getConnHandle() != connHandle) {
|
if (pClient->getConnHandle() != connHandle) {
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -904,6 +925,7 @@ bool NimBLEClient::exchangeMTU() {
|
||||||
int NimBLEClient::handleGapEvent(struct ble_gap_event* event, void* arg) {
|
int NimBLEClient::handleGapEvent(struct ble_gap_event* event, void* arg) {
|
||||||
NimBLEClient* pClient = (NimBLEClient*)arg;
|
NimBLEClient* pClient = (NimBLEClient*)arg;
|
||||||
int rc = 0;
|
int rc = 0;
|
||||||
|
NimBLETaskData* pTaskData = pClient->m_pTaskData; // save a copy in case client is deleted
|
||||||
|
|
||||||
NIMBLE_LOGD(LOG_TAG, "Got Client event %s", NimBLEUtils::gapEventToString(event->type));
|
NIMBLE_LOGD(LOG_TAG, "Got Client event %s", NimBLEUtils::gapEventToString(event->type));
|
||||||
|
|
||||||
|
@ -917,7 +939,7 @@ int NimBLEClient::handleGapEvent(struct ble_gap_event* event, void* arg) {
|
||||||
|
|
||||||
rc = event->disconnect.reason;
|
rc = event->disconnect.reason;
|
||||||
// If Host reset tell the device now before returning to prevent
|
// If Host reset tell the device now before returning to prevent
|
||||||
// any errors caused by calling host functions before resyncing.
|
// any errors caused by calling host functions before re-syncing.
|
||||||
switch (rc) {
|
switch (rc) {
|
||||||
case BLE_HS_ECONTROLLER:
|
case BLE_HS_ECONTROLLER:
|
||||||
case BLE_HS_ETIMEOUT_HCI:
|
case BLE_HS_ETIMEOUT_HCI:
|
||||||
|
@ -930,28 +952,35 @@ int NimBLEClient::handleGapEvent(struct ble_gap_event* event, void* arg) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
NIMBLE_LOGD(LOG_TAG, "disconnect; reason=%d, %s", rc, NimBLEUtils::returnCodeToString(rc));
|
||||||
|
|
||||||
pClient->m_terminateFailCount = 0;
|
pClient->m_terminateFailCount = 0;
|
||||||
NimBLEDevice::removeIgnored(pClient->m_peerAddress);
|
NimBLEDevice::removeIgnored(pClient->m_peerAddress);
|
||||||
pClient->m_connHandle = BLE_HS_CONN_HANDLE_NONE;
|
|
||||||
|
|
||||||
// If we received a connected event but did not get established
|
// Don't call the disconnect callback if we are waiting for a connection to complete and it fails
|
||||||
// then a disconnect event will be sent but we should not send it to the
|
if (rc != (BLE_HS_ERR_HCI_BASE + BLE_ERR_CONN_ESTABLISHMENT) || pClient->m_config.asyncConnect) {
|
||||||
// app for processing. Instead we will ensure the task is released
|
pClient->m_pClientCallbacks->onDisconnect(pClient, rc);
|
||||||
// and report the error.
|
|
||||||
if (!pClient->m_connEstablished) {
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
NIMBLE_LOGI(LOG_TAG, "disconnect; reason=%d, %s", rc, NimBLEUtils::returnCodeToString(rc));
|
pClient->m_connHandle = BLE_HS_CONN_HANDLE_NONE;
|
||||||
|
|
||||||
|
if (pClient->m_config.deleteOnDisconnect) {
|
||||||
|
// If we are set to self delete on disconnect but we have a task waiting on the connection
|
||||||
|
// completion we will set the flag to delete on connect fail instead of deleting here
|
||||||
|
// to prevent segmentation faults or double deleting
|
||||||
|
if (pTaskData != nullptr && rc == (BLE_HS_ERR_HCI_BASE + BLE_ERR_CONN_ESTABLISHMENT)) {
|
||||||
|
pClient->m_config.deleteOnConnectFail = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
NimBLEDevice::deleteClient(pClient);
|
||||||
|
}
|
||||||
|
|
||||||
pClient->m_connEstablished = false;
|
|
||||||
pClient->m_pClientCallbacks->onDisconnect(pClient, rc);
|
|
||||||
break;
|
break;
|
||||||
} // BLE_GAP_EVENT_DISCONNECT
|
} // BLE_GAP_EVENT_DISCONNECT
|
||||||
|
|
||||||
case BLE_GAP_EVENT_CONNECT: {
|
case BLE_GAP_EVENT_CONNECT: {
|
||||||
// If we aren't waiting for this connection response we should drop the connection immediately.
|
// If we aren't waiting for this connection response we should drop the connection immediately.
|
||||||
if (pClient->isConnected() || (!pClient->m_asyncConnect && pClient->m_pTaskData == nullptr)) {
|
if (pClient->isConnected() || (!pClient->m_config.asyncConnect && pClient->m_pTaskData == nullptr)) {
|
||||||
ble_gap_terminate(event->connect.conn_handle, BLE_ERR_REM_USER_CONN_TERM);
|
ble_gap_terminate(event->connect.conn_handle, BLE_ERR_REM_USER_CONN_TERM);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -959,12 +988,10 @@ int NimBLEClient::handleGapEvent(struct ble_gap_event* event, void* arg) {
|
||||||
NimBLEDevice::setConnectionInProgress(false);
|
NimBLEDevice::setConnectionInProgress(false);
|
||||||
rc = event->connect.status;
|
rc = event->connect.status;
|
||||||
if (rc == 0) {
|
if (rc == 0) {
|
||||||
NIMBLE_LOGI(LOG_TAG, "Connected event");
|
|
||||||
|
|
||||||
pClient->m_connHandle = event->connect.conn_handle;
|
pClient->m_connHandle = event->connect.conn_handle;
|
||||||
if (pClient->m_exchangeMTU) {
|
if (pClient->m_config.exchangeMTU) {
|
||||||
if (!pClient->exchangeMTU() && !pClient->m_asyncConnect) {
|
if (!pClient->exchangeMTU() && !pClient->m_config.asyncConnect) {
|
||||||
rc = pClient->m_lastErr;
|
rc = pClient->m_lastErr; // sets the error in the task data
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -974,15 +1001,19 @@ int NimBLEClient::handleGapEvent(struct ble_gap_event* event, void* arg) {
|
||||||
NimBLEDevice::addIgnored(pClient->m_peerAddress);
|
NimBLEDevice::addIgnored(pClient->m_peerAddress);
|
||||||
} else {
|
} else {
|
||||||
pClient->m_connHandle = BLE_HS_CONN_HANDLE_NONE;
|
pClient->m_connHandle = BLE_HS_CONN_HANDLE_NONE;
|
||||||
if (!pClient->m_asyncConnect) {
|
if (!pClient->m_config.asyncConnect) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (pClient->m_config.deleteOnConnectFail) { // async connect
|
||||||
|
delete pClient;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pClient->m_asyncConnect) {
|
if (pClient->m_config.asyncConnect) {
|
||||||
pClient->m_connEstablished = rc == 0;
|
|
||||||
pClient->m_pClientCallbacks->onConnect(pClient);
|
pClient->m_pClientCallbacks->onConnect(pClient);
|
||||||
} else if (!pClient->m_exchangeMTU) {
|
} else if (!pClient->m_config.exchangeMTU) {
|
||||||
break; // not waiting for MTU exchange so release the task now.
|
break; // not waiting for MTU exchange so release the task now.
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1005,13 +1036,6 @@ int NimBLEClient::handleGapEvent(struct ble_gap_event* event, void* arg) {
|
||||||
|
|
||||||
case BLE_GAP_EVENT_NOTIFY_RX: {
|
case BLE_GAP_EVENT_NOTIFY_RX: {
|
||||||
if (pClient->m_connHandle != event->notify_rx.conn_handle) return 0;
|
if (pClient->m_connHandle != event->notify_rx.conn_handle) return 0;
|
||||||
|
|
||||||
// If a notification comes before this flag is set we might
|
|
||||||
// access a vector while it is being cleared in connect()
|
|
||||||
if (!pClient->m_connEstablished) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
NIMBLE_LOGD(LOG_TAG, "Notify Received for handle: %d", event->notify_rx.attr_handle);
|
NIMBLE_LOGD(LOG_TAG, "Notify Received for handle: %d", event->notify_rx.attr_handle);
|
||||||
|
|
||||||
for (const auto& svc : pClient->m_svcVec) {
|
for (const auto& svc : pClient->m_svcVec) {
|
||||||
|
@ -1170,8 +1194,8 @@ int NimBLEClient::handleGapEvent(struct ble_gap_event* event, void* arg) {
|
||||||
}
|
}
|
||||||
} // Switch
|
} // Switch
|
||||||
|
|
||||||
if (pClient->m_pTaskData != nullptr) {
|
if (pTaskData != nullptr) {
|
||||||
NimBLEUtils::taskRelease(*pClient->m_pTaskData, rc);
|
NimBLEUtils::taskRelease(*pTaskData, rc);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1193,10 +1217,10 @@ bool NimBLEClient::isConnected() const {
|
||||||
void NimBLEClient::setClientCallbacks(NimBLEClientCallbacks* pClientCallbacks, bool deleteCallbacks) {
|
void NimBLEClient::setClientCallbacks(NimBLEClientCallbacks* pClientCallbacks, bool deleteCallbacks) {
|
||||||
if (pClientCallbacks != nullptr) {
|
if (pClientCallbacks != nullptr) {
|
||||||
m_pClientCallbacks = pClientCallbacks;
|
m_pClientCallbacks = pClientCallbacks;
|
||||||
m_deleteCallbacks = deleteCallbacks;
|
m_config.deleteCallbacks = deleteCallbacks;
|
||||||
} else {
|
} else {
|
||||||
m_pClientCallbacks = &defaultCallbacks;
|
m_pClientCallbacks = &defaultCallbacks;
|
||||||
m_deleteCallbacks = false;
|
m_config.deleteCallbacks = false;
|
||||||
}
|
}
|
||||||
} // setClientCallbacks
|
} // setClientCallbacks
|
||||||
|
|
||||||
|
|
|
@ -51,7 +51,8 @@ class NimBLEClient {
|
||||||
bool connect(const NimBLEAddress& address, bool deleteAttributes = true, bool asyncConnect = false, bool exchangeMTU = true);
|
bool connect(const NimBLEAddress& address, bool deleteAttributes = true, bool asyncConnect = false, bool exchangeMTU = true);
|
||||||
bool connect(bool deleteAttributes = true, bool asyncConnect = false, bool exchangeMTU = true);
|
bool connect(bool deleteAttributes = true, bool asyncConnect = false, bool exchangeMTU = true);
|
||||||
bool disconnect(uint8_t reason = BLE_ERR_REM_USER_CONN_TERM);
|
bool disconnect(uint8_t reason = BLE_ERR_REM_USER_CONN_TERM);
|
||||||
bool cancelConnect();
|
bool cancelConnect() const;
|
||||||
|
void setSelfDelete(bool deleteOnDisconnect, bool deleteOnConnectFail);
|
||||||
NimBLEAddress getPeerAddress() const;
|
NimBLEAddress getPeerAddress() const;
|
||||||
bool setPeerAddress(const NimBLEAddress& address);
|
bool setPeerAddress(const NimBLEAddress& address);
|
||||||
int getRssi() const;
|
int getRssi() const;
|
||||||
|
@ -95,6 +96,17 @@ class NimBLEClient {
|
||||||
void setConnectPhy(uint8_t mask);
|
void setConnectPhy(uint8_t mask);
|
||||||
# endif
|
# endif
|
||||||
|
|
||||||
|
struct Config {
|
||||||
|
uint8_t deleteCallbacks : 1; // Delete the callback object when the client is deleted.
|
||||||
|
uint8_t deleteOnDisconnect : 1; // Delete the client when disconnected.
|
||||||
|
uint8_t deleteOnConnectFail : 1; // Delete the client when a connection attempt fails.
|
||||||
|
uint8_t asyncConnect : 1; // Connect asynchronously.
|
||||||
|
uint8_t exchangeMTU : 1; // Exchange MTU after connection.
|
||||||
|
};
|
||||||
|
|
||||||
|
Config getConfig() const;
|
||||||
|
void setConfig(Config config);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
NimBLEClient(const NimBLEAddress& peerAddress);
|
NimBLEClient(const NimBLEAddress& peerAddress);
|
||||||
~NimBLEClient();
|
~NimBLEClient();
|
||||||
|
@ -117,10 +129,8 @@ class NimBLEClient {
|
||||||
NimBLEClientCallbacks* m_pClientCallbacks;
|
NimBLEClientCallbacks* m_pClientCallbacks;
|
||||||
uint16_t m_connHandle;
|
uint16_t m_connHandle;
|
||||||
uint8_t m_terminateFailCount;
|
uint8_t m_terminateFailCount;
|
||||||
bool m_deleteCallbacks;
|
Config m_config;
|
||||||
bool m_connEstablished;
|
|
||||||
bool m_asyncConnect;
|
|
||||||
bool m_exchangeMTU;
|
|
||||||
# if CONFIG_BT_NIMBLE_EXT_ADV
|
# if CONFIG_BT_NIMBLE_EXT_ADV
|
||||||
uint8_t m_phyMask;
|
uint8_t m_phyMask;
|
||||||
# endif
|
# endif
|
||||||
|
|
|
@ -322,41 +322,28 @@ bool NimBLEDevice::deleteClient(NimBLEClient* pClient) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set the connection established flag to false to stop notifications
|
|
||||||
// from accessing the attribute vectors while they are being deleted.
|
|
||||||
pClient->m_connEstablished = false;
|
|
||||||
int rc = 0;
|
|
||||||
|
|
||||||
if (pClient->isConnected()) {
|
|
||||||
if (!pClient->disconnect()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (pClient->isConnected()) {
|
|
||||||
ble_npl_time_delay(1);
|
|
||||||
}
|
|
||||||
// Since we set the flag to false the app will not get a callback
|
|
||||||
// in the disconnect event so we call it here for good measure.
|
|
||||||
pClient->m_pClientCallbacks->onDisconnect(pClient, BLE_ERR_CONN_TERM_LOCAL);
|
|
||||||
} else if (pClient->m_pTaskData != nullptr) {
|
|
||||||
rc = ble_gap_conn_cancel();
|
|
||||||
if (rc != 0 && rc != BLE_HS_EALREADY) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (pClient->m_pTaskData != nullptr) {
|
|
||||||
ble_npl_time_delay(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto& clt : m_pClients) {
|
for (auto& clt : m_pClients) {
|
||||||
if (clt == pClient) {
|
if (clt == pClient) {
|
||||||
delete pClient;
|
if (clt->isConnected()) {
|
||||||
clt = nullptr;
|
clt->m_config.deleteOnDisconnect = true;
|
||||||
|
if (!clt->disconnect()) {
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
} else if (pClient->m_pTaskData != nullptr) {
|
||||||
|
clt->m_config.deleteOnConnectFail = true;
|
||||||
|
if (!clt->cancelConnect()) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
delete clt;
|
||||||
|
clt = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
} // deleteClient
|
} // deleteClient
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -69,6 +69,12 @@ int NimBLERemoteCharacteristic::descriptorDiscCB(
|
||||||
const auto pChr = (NimBLERemoteCharacteristic*)pTaskData->m_pInstance;
|
const auto pChr = (NimBLERemoteCharacteristic*)pTaskData->m_pInstance;
|
||||||
const NimBLEUUID* uuidFilter = filter->uuid;
|
const NimBLEUUID* uuidFilter = filter->uuid;
|
||||||
|
|
||||||
|
if (error->status == BLE_HS_ENOTCONN) {
|
||||||
|
NIMBLE_LOGE(LOG_TAG, "<< Descriptor Discovery; Not connected");
|
||||||
|
NimBLEUtils::taskRelease(*pTaskData, error->status);
|
||||||
|
return error->status;
|
||||||
|
}
|
||||||
|
|
||||||
if (pChr->getHandle() != chr_val_handle) {
|
if (pChr->getHandle() != chr_val_handle) {
|
||||||
rc = BLE_HS_EDONE; // descriptor not for this characteristic
|
rc = BLE_HS_EDONE; // descriptor not for this characteristic
|
||||||
}
|
}
|
||||||
|
|
|
@ -148,6 +148,13 @@ int NimBLERemoteService::characteristicDiscCB(uint16_t conn_handle,
|
||||||
auto pTaskData = (NimBLETaskData*)arg;
|
auto pTaskData = (NimBLETaskData*)arg;
|
||||||
const auto pSvc = (NimBLERemoteService*)pTaskData->m_pInstance;
|
const auto pSvc = (NimBLERemoteService*)pTaskData->m_pInstance;
|
||||||
|
|
||||||
|
|
||||||
|
if (error->status == BLE_HS_ENOTCONN) {
|
||||||
|
NIMBLE_LOGE(LOG_TAG, "<< Characteristic Discovery; Not connected");
|
||||||
|
NimBLEUtils::taskRelease(*pTaskData, error->status);
|
||||||
|
return error->status;
|
||||||
|
}
|
||||||
|
|
||||||
// Make sure the discovery is for this device
|
// Make sure the discovery is for this device
|
||||||
if (pSvc->getClient()->getConnHandle() != conn_handle) {
|
if (pSvc->getClient()->getConnHandle() != conn_handle) {
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -91,6 +91,12 @@ int NimBLERemoteValueAttribute::onWriteCB(uint16_t conn_handle, const ble_gatt_e
|
||||||
auto pTaskData = static_cast<NimBLETaskData*>(arg);
|
auto pTaskData = static_cast<NimBLETaskData*>(arg);
|
||||||
const auto pAtt = static_cast<NimBLERemoteValueAttribute*>(pTaskData->m_pInstance);
|
const auto pAtt = static_cast<NimBLERemoteValueAttribute*>(pTaskData->m_pInstance);
|
||||||
|
|
||||||
|
if (error->status == BLE_HS_ENOTCONN) {
|
||||||
|
NIMBLE_LOGE(LOG_TAG, "<< Write complete; Not connected");
|
||||||
|
NimBLEUtils::taskRelease(*pTaskData, error->status);
|
||||||
|
return error->status;
|
||||||
|
}
|
||||||
|
|
||||||
if (pAtt->getClient()->getConnHandle() != conn_handle) {
|
if (pAtt->getClient()->getConnHandle() != conn_handle) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -170,6 +176,12 @@ int NimBLERemoteValueAttribute::onReadCB(uint16_t conn_handle, const ble_gatt_er
|
||||||
auto pTaskData = static_cast<NimBLETaskData*>(arg);
|
auto pTaskData = static_cast<NimBLETaskData*>(arg);
|
||||||
const auto pAtt = static_cast<NimBLERemoteValueAttribute*>(pTaskData->m_pInstance);
|
const auto pAtt = static_cast<NimBLERemoteValueAttribute*>(pTaskData->m_pInstance);
|
||||||
|
|
||||||
|
if (error->status == BLE_HS_ENOTCONN) {
|
||||||
|
NIMBLE_LOGE(LOG_TAG, "<< Read complete; Not connected");
|
||||||
|
NimBLEUtils::taskRelease(*pTaskData, error->status);
|
||||||
|
return error->status;
|
||||||
|
}
|
||||||
|
|
||||||
if (pAtt->getClient()->getConnHandle() != conn_handle) {
|
if (pAtt->getClient()->getConnHandle() != conn_handle) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -143,7 +143,7 @@ const char* NimBLEUtils::returnCodeToString(int rc) {
|
||||||
case BLE_HS_ENOTSUP:
|
case BLE_HS_ENOTSUP:
|
||||||
return "Operation disabled at compile time";
|
return "Operation disabled at compile time";
|
||||||
case BLE_HS_EAPP:
|
case BLE_HS_EAPP:
|
||||||
return "Application error";
|
return "Operation canceled by app";
|
||||||
case BLE_HS_EBADDATA:
|
case BLE_HS_EBADDATA:
|
||||||
return "Invalid command from peer";
|
return "Invalid command from peer";
|
||||||
case BLE_HS_EOS:
|
case BLE_HS_EOS:
|
||||||
|
|
Loading…
Reference in a new issue