| /* |
| * |
| * Copyright (c) 2020 Project CHIP Authors |
| * Copyright (c) 2014-2017 Nest Labs, Inc. |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| /** |
| * @file |
| * This file implements a Bluetooth Low Energy (BLE) connection |
| * endpoint abstraction for the byte-streaming, |
| * connection-oriented CHIP over Bluetooth Low Energy (CHIPoBLE) |
| * Bluetooth Transport Protocol (BTP). |
| * |
| */ |
| |
| #ifndef __STDC_LIMIT_MACROS |
| #define __STDC_LIMIT_MACROS |
| #endif |
| #include <stdint.h> |
| #include <string.h> |
| |
| #include <ble/BleConfig.h> |
| |
| #if CONFIG_NETWORK_LAYER_BLE |
| #include <core/CHIPConfig.h> |
| |
| #include <support/CHIPFaultInjection.h> |
| #include <support/CodeUtils.h> |
| #include <support/FlagUtils.hpp> |
| #include <support/logging/CHIPLogging.h> |
| |
| #include <ble/BLEEndPoint.h> |
| #include <ble/BleLayer.h> |
| #include <ble/BtpEngine.h> |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| #include "BtpEngineTest.h" |
| #endif |
| |
| // clang-format off |
| |
| // Define below to enable extremely verbose, BLE end point-specific debug logging. |
| #undef CHIP_BLE_END_POINT_DEBUG_LOGGING_ENABLED |
| |
| #ifdef CHIP_BLE_END_POINT_DEBUG_LOGGING_ENABLED |
| #define ChipLogDebugBleEndPoint(MOD, MSG, ...) ChipLogError(MOD, MSG, ## __VA_ARGS__) |
| #else |
| #define ChipLogDebugBleEndPoint(MOD, MSG, ...) |
| #endif |
| |
| /** |
| * @def BLE_CONFIG_IMMEDIATE_ACK_WINDOW_THRESHOLD |
| * |
| * @brief |
| * If an end point's receive window drops equal to or below this value, it will send an immediate acknowledgement |
| * packet to re-open its window instead of waiting for the send-ack timer to expire. |
| * |
| */ |
| #define BLE_CONFIG_IMMEDIATE_ACK_WINDOW_THRESHOLD 1 |
| |
| /** |
| * @def BLE_CONNECT_TIMEOUT_MS |
| * |
| * @brief |
| * This is the amount of time, in milliseconds, after a BLE end point initiates a transport protocol connection |
| * or receives the initial portion of a connect request before the end point will automatically release its BLE |
| * connection and free itself if the transport connection has not been established. |
| * |
| */ |
| #define BLE_CONNECT_TIMEOUT_MS 5000 // 5 seconds |
| |
| /** |
| * @def BLE_UNSUBSCRIBE_TIMEOUT_MS |
| * |
| * @brief |
| * This is amount of time, in milliseconds, which a BLE end point will wait for an unsubscribe operation to complete |
| * before it automatically releases its BLE connection and frees itself. The default value of 5 seconds is arbitary. |
| * |
| */ |
| #define BLE_UNSUBSCRIBE_TIMEOUT_MS 5000 // 5 seconds |
| |
| #define BTP_ACK_RECEIVED_TIMEOUT_MS 15000 // 15 seconds |
| #define BTP_ACK_SEND_TIMEOUT_MS 2500 // 2.5 seconds |
| |
| #define BTP_WINDOW_NO_ACK_SEND_THRESHOLD 1 // Data fragments may only be sent without piggybacked |
| // acks if receiver's window size is above this threshold. |
| |
| // clang-format on |
| |
| namespace chip { |
| namespace Ble { |
| |
| BLE_ERROR BLEEndPoint::StartConnect() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| BleTransportCapabilitiesRequestMessage req; |
| PacketBuffer * buf = nullptr; |
| int i; |
| int numVersions; |
| |
| // Ensure we're in the correct state. |
| VerifyOrExit(mState == kState_Ready, err = BLE_ERROR_INCORRECT_STATE); |
| mState = kState_Connecting; |
| |
| // Build BLE transport protocol capabilities request. |
| buf = PacketBuffer::New(); |
| VerifyOrExit(buf != nullptr, err = BLE_ERROR_NO_MEMORY); |
| |
| // Zero-initialize BLE transport capabilities request. |
| memset(&req, 0, sizeof(req)); |
| |
| req.mMtu = mBle->mPlatformDelegate->GetMTU(mConnObj); |
| |
| req.mWindowSize = BLE_MAX_RECEIVE_WINDOW_SIZE; |
| |
| // Populate request with highest supported protocol versions |
| numVersions = CHIP_BLE_TRANSPORT_PROTOCOL_MAX_SUPPORTED_VERSION - CHIP_BLE_TRANSPORT_PROTOCOL_MIN_SUPPORTED_VERSION + 1; |
| VerifyOrExit(numVersions <= NUM_SUPPORTED_PROTOCOL_VERSIONS, err = BLE_ERROR_INCOMPATIBLE_PROTOCOL_VERSIONS); |
| for (i = 0; i < numVersions; i++) |
| { |
| req.SetSupportedProtocolVersion(i, CHIP_BLE_TRANSPORT_PROTOCOL_MAX_SUPPORTED_VERSION - i); |
| } |
| |
| err = req.Encode(buf); |
| SuccessOrExit(err); |
| |
| // Start connect timer. Canceled when end point freed or connection established. |
| err = StartConnectTimer(); |
| SuccessOrExit(err); |
| |
| // Send BLE transport capabilities request to peripheral via GATT write. |
| if (!SendWrite(buf)) |
| { |
| err = BLE_ERROR_GATT_WRITE_FAILED; |
| ExitNow(); |
| } |
| |
| // Free request buffer on write confirmation. Stash a reference to it in mSendQueue, which we don't use anyway |
| // until the connection has been set up. |
| QueueTx(buf, kType_Data); |
| buf = nullptr; |
| |
| exit: |
| if (buf != nullptr) |
| { |
| PacketBuffer::Free(buf); |
| } |
| |
| // If we failed to initiate the connection, close the end point. |
| if (err != BLE_NO_ERROR) |
| { |
| StopConnectTimer(); |
| DoClose(kBleCloseFlag_AbortTransmission, err); |
| } |
| |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::HandleConnectComplete() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| |
| mState = kState_Connected; |
| |
| // Cancel the connect timer. |
| StopConnectTimer(); |
| |
| // We've successfully completed the BLE transport protocol handshake, so let the application know we're open for business. |
| if (OnConnectComplete != nullptr) |
| { |
| // Indicate connect complete to next-higher layer. |
| OnConnectComplete(this, BLE_NO_ERROR); |
| } |
| else |
| { |
| // If no connect complete callback has been set up, close the end point. |
| err = BLE_ERROR_NO_CONNECT_COMPLETE_CALLBACK; |
| } |
| |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::HandleReceiveConnectionComplete() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| |
| ChipLogDebugBleEndPoint(Ble, "entered HandleReceiveConnectionComplete"); |
| mState = kState_Connected; |
| |
| // Cancel receive connection timer. |
| StopReceiveConnectionTimer(); |
| |
| // We've successfully completed the BLE transport protocol handshake, so let the application know we're open for business. |
| if (mBle->OnChipBleConnectReceived != nullptr) |
| { |
| // Indicate BLE transport protocol connection received to next-higher layer. |
| mBle->OnChipBleConnectReceived(this); |
| } |
| else |
| { |
| err = BLE_ERROR_NO_CONNECTION_RECEIVED_CALLBACK; |
| } |
| |
| return err; |
| } |
| |
| void BLEEndPoint::HandleSubscribeReceived() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| |
| VerifyOrExit(mState == kState_Connecting || mState == kState_Aborting, err = BLE_ERROR_INCORRECT_STATE); |
| VerifyOrExit(mSendQueue != nullptr, err = BLE_ERROR_INCORRECT_STATE); |
| |
| // Send BTP capabilities response to peripheral via GATT indication. |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| VerifyOrExit(mBtpEngine.PopPacketTag(mSendQueue) == kType_Data, err = BLE_ERROR_INVALID_BTP_HEADER_FLAGS); |
| #endif |
| if (!SendIndication(mSendQueue)) |
| { |
| // Ensure transmit queue is empty and set to NULL. |
| QueueTxLock(); |
| PacketBuffer::Free(mSendQueue); |
| mSendQueue = nullptr; |
| QueueTxUnlock(); |
| |
| ChipLogError(Ble, "cap resp ind failed"); |
| err = BLE_ERROR_GATT_INDICATE_FAILED; |
| ExitNow(); |
| } |
| |
| // Shrink remote receive window counter by 1, since we've sent an indication which requires acknowledgement. |
| mRemoteReceiveWindowSize -= 1; |
| ChipLogDebugBleEndPoint(Ble, "decremented remote rx window, new size = %u", mRemoteReceiveWindowSize); |
| |
| // Start ack recvd timer for handshake indication. |
| err = StartAckReceivedTimer(); |
| SuccessOrExit(err); |
| |
| ChipLogDebugBleEndPoint(Ble, "got subscribe, sent indication w/ capabilities response"); |
| |
| // If SendIndication returns true, mSendQueue is freed on indication confirmation, or on close in case of |
| // connection error. |
| |
| if (mState != kState_Aborting) |
| { |
| // If peripheral accepted the BTP connection, its end point must enter the connected state here, i.e. before it |
| // receives a GATT confirmation for the capabilities response indication. This behavior is required to handle the |
| // case where a peripheral's BLE controller passes up the central's first message fragment write before the |
| // capabilities response indication confirmation. If the end point waited for this indication confirmation before |
| // it entered the connected state, it'd be in the wrong state to receive the central's first data write, and drop |
| // the corresponding message fragment. |
| err = HandleReceiveConnectionComplete(); |
| SuccessOrExit(err); |
| } // Else State == kState_Aborting, so we'll close end point when indication confirmation received. |
| |
| exit: |
| if (err != BLE_NO_ERROR) |
| { |
| DoClose(kBleCloseFlag_SuppressCallback | kBleCloseFlag_AbortTransmission, err); |
| } |
| |
| return; |
| } |
| |
| void BLEEndPoint::HandleSubscribeComplete() |
| { |
| ChipLogProgress(Ble, "subscribe complete, ep = %p", this); |
| SetFlag(mConnStateFlags, kConnState_GattOperationInFlight, false); |
| |
| BLE_ERROR err = DriveSending(); |
| |
| if (err != BLE_NO_ERROR) |
| { |
| DoClose(kBleCloseFlag_AbortTransmission, BLE_NO_ERROR); |
| } |
| } |
| |
| void BLEEndPoint::HandleUnsubscribeComplete() |
| { |
| // Don't bother to clear GattOperationInFlight, we're about to free the end point anyway. |
| Free(); |
| } |
| |
| bool BLEEndPoint::IsConnected(uint8_t state) const |
| { |
| return (state == kState_Connected || state == kState_Closing); |
| } |
| |
| bool BLEEndPoint::IsUnsubscribePending() const |
| { |
| return (GetFlag(mTimerStateFlags, kTimerState_UnsubscribeTimerRunning)); |
| } |
| |
| void BLEEndPoint::Abort() |
| { |
| // No more callbacks after this point, since application explicitly called Abort(). |
| OnConnectComplete = nullptr; |
| OnConnectionClosed = nullptr; |
| OnMessageReceived = nullptr; |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| OnCommandReceived = NULL; |
| #endif |
| |
| DoClose(kBleCloseFlag_SuppressCallback | kBleCloseFlag_AbortTransmission, BLE_NO_ERROR); |
| } |
| |
| void BLEEndPoint::Close() |
| { |
| // No more callbacks after this point, since application explicitly called Close(). |
| OnConnectComplete = nullptr; |
| OnConnectionClosed = nullptr; |
| OnMessageReceived = nullptr; |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| OnCommandReceived = NULL; |
| #endif |
| |
| DoClose(kBleCloseFlag_SuppressCallback, BLE_NO_ERROR); |
| } |
| |
| void BLEEndPoint::DoClose(uint8_t flags, BLE_ERROR err) |
| { |
| uint8_t oldState = mState; |
| |
| // If end point is not closed or closing, OR end point was closing gracefully, but tx abort has been specified... |
| if ((mState != kState_Closed && mState != kState_Closing) || |
| (mState == kState_Closing && (flags & kBleCloseFlag_AbortTransmission))) |
| { |
| // Cancel Connect and ReceiveConnect timers if they are running. |
| // Check role first to avoid needless iteration over timer pool. |
| if (mRole == kBleRole_Central) |
| { |
| StopConnectTimer(); |
| } |
| else // (mRole == kBleRole_Peripheral), verified on Init |
| { |
| StopReceiveConnectionTimer(); |
| } |
| |
| // If transmit buffer is empty or a transmission abort was specified... |
| if (mBtpEngine.TxState() == BtpEngine::kState_Idle || (flags & kBleCloseFlag_AbortTransmission)) |
| { |
| FinalizeClose(oldState, flags, err); |
| } |
| else |
| { |
| // Wait for send queue and fragmenter's tx buffer to become empty, to ensure all pending messages have been |
| // sent. Only free end point and tell platform it can throw away the underlying BLE connection once all |
| // pending messages have been sent and acknowledged by the remote CHIPoBLE stack, or once the remote stack |
| // closes the CHIPoBLE connection. |
| // |
| // In so doing, BLEEndPoint attempts to emulate the level of reliability afforded by TCPEndPoint and TCP |
| // sockets in general with a typical default SO_LINGER option. That said, there is no hard guarantee that |
| // pending messages will be sent once (Do)Close() is called, so developers should use application-level |
| // messages to confirm the receipt of all data sent prior to a Close() call. |
| mState = kState_Closing; |
| |
| if ((flags & kBleCloseFlag_SuppressCallback) == 0) |
| { |
| DoCloseCallback(oldState, flags, err); |
| } |
| } |
| } |
| } |
| |
| void BLEEndPoint::FinalizeClose(uint8_t oldState, uint8_t flags, BLE_ERROR err) |
| { |
| mState = kState_Closed; |
| |
| // Ensure transmit queue is empty and set to NULL. |
| QueueTxLock(); |
| PacketBuffer::Free(mSendQueue); |
| mSendQueue = nullptr; |
| QueueTxUnlock(); |
| |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| PacketBuffer::Free(mBtpEngineTest.mCommandReceiveQueue); |
| mBtpEngineTest.mCommandReceiveQueue = NULL; |
| #endif |
| |
| // Fire application's close callback if we haven't already, and it's not suppressed. |
| if (oldState != kState_Closing && (flags & kBleCloseFlag_SuppressCallback) == 0) |
| { |
| DoCloseCallback(oldState, flags, err); |
| } |
| |
| // If underlying BLE connection has closed, connection object is invalid, so just free the end point and return. |
| if (err == BLE_ERROR_REMOTE_DEVICE_DISCONNECTED || err == BLE_ERROR_APP_CLOSED_CONNECTION) |
| { |
| mConnObj = BLE_CONNECTION_UNINITIALIZED; // Clear handle to BLE connection, so we don't double-close it. |
| Free(); |
| } |
| else // Otherwise, try to signal close to remote device before end point releases BLE connection and frees itself. |
| { |
| if (mRole == kBleRole_Central && GetFlag(mConnStateFlags, kConnState_DidBeginSubscribe)) |
| { |
| // Cancel send and receive-ack timers, if running. |
| StopAckReceivedTimer(); |
| StopSendAckTimer(); |
| |
| // Indicate close of chipConnection to peripheral via GATT unsubscribe. Keep end point allocated until |
| // unsubscribe completes or times out, so platform doesn't close underlying BLE connection before |
| // we're really sure the unsubscribe request has been sent. |
| if (!mBle->mPlatformDelegate->UnsubscribeCharacteristic(mConnObj, &CHIP_BLE_SVC_ID, &mBle->CHIP_BLE_CHAR_2_ID)) |
| { |
| ChipLogError(Ble, "BtpEngine unsub failed"); |
| |
| // If unsubscribe fails, release BLE connection and free end point immediately. |
| Free(); |
| } |
| else if (mConnObj != BLE_CONNECTION_UNINITIALIZED) |
| { |
| // Unsubscribe request was sent successfully, and a confirmation wasn't spontaneously generated or |
| // received in the downcall to UnsubscribeCharacteristic, so set timer for the unsubscribe to complete. |
| err = StartUnsubscribeTimer(); |
| |
| if (err != BLE_NO_ERROR) |
| { |
| Free(); |
| } |
| |
| // Mark unsubscribe GATT operation in progress. |
| SetFlag(mConnStateFlags, kConnState_GattOperationInFlight, true); |
| } |
| } |
| else // mRole == kBleRole_Peripheral, OR GetFlag(mTimerStateFlags, kConnState_DidBeginSubscribe) == false... |
| { |
| Free(); |
| } |
| } |
| } |
| |
| void BLEEndPoint::DoCloseCallback(uint8_t state, uint8_t flags, BLE_ERROR err) |
| { |
| if (state == kState_Connecting) |
| { |
| if (OnConnectComplete != nullptr) |
| { |
| OnConnectComplete(this, err); |
| } |
| } |
| else |
| { |
| if (OnConnectionClosed != nullptr) |
| { |
| OnConnectionClosed(this, err); |
| } |
| } |
| |
| // Callback fires once per end point lifetime. |
| OnConnectComplete = nullptr; |
| OnConnectionClosed = nullptr; |
| } |
| |
| void BLEEndPoint::ReleaseBleConnection() |
| { |
| if (mConnObj != BLE_CONNECTION_UNINITIALIZED) |
| { |
| if (GetFlag(mConnStateFlags, kConnState_AutoClose)) |
| { |
| ChipLogProgress(Ble, "Auto-closing end point's BLE connection."); |
| mBle->mPlatformDelegate->CloseConnection(mConnObj); |
| } |
| else |
| { |
| ChipLogProgress(Ble, "Releasing end point's BLE connection back to application."); |
| mBle->mApplicationDelegate->NotifyChipConnectionClosed(mConnObj); |
| } |
| |
| // Never release the same BLE connection twice. |
| mConnObj = BLE_CONNECTION_UNINITIALIZED; |
| } |
| } |
| |
| void BLEEndPoint::Free() |
| { |
| // Release BLE connection. Will close connection if AutoClose enabled for this end point. Otherwise, informs |
| // application that CHIP is done with this BLE connection, and application makes decision about whether to close |
| // and clean up or retain connection. |
| ReleaseBleConnection(); |
| |
| // Clear fragmentation and reassembly engine's Tx and Rx buffers. Counters will be reset by next engine init. |
| FreeBtpEngine(); |
| |
| // Clear pending ack buffer, if any. |
| PacketBuffer::Free(mAckToSend); |
| |
| // Cancel all timers. |
| StopConnectTimer(); |
| StopReceiveConnectionTimer(); |
| StopAckReceivedTimer(); |
| StopSendAckTimer(); |
| StopUnsubscribeTimer(); |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| mBtpEngineTest.StopTestTimer(); |
| // Clear callback |
| OnCommandReceived = NULL; |
| #endif |
| |
| // Clear callbacks. |
| OnConnectComplete = nullptr; |
| OnMessageReceived = nullptr; |
| OnConnectionClosed = nullptr; |
| |
| // Clear handle to underlying BLE connection. |
| mConnObj = BLE_CONNECTION_UNINITIALIZED; |
| |
| // Release the AddRef() that happened when the end point was allocated. |
| Release(); |
| } |
| |
| void BLEEndPoint::FreeBtpEngine() |
| { |
| PacketBuffer * buf; |
| |
| // Free transmit disassembly buffer |
| buf = mBtpEngine.TxPacket(); |
| mBtpEngine.ClearTxPacket(); |
| PacketBuffer::Free(buf); |
| |
| // Free receive reassembly buffer |
| buf = mBtpEngine.RxPacket(); |
| mBtpEngine.ClearRxPacket(); |
| PacketBuffer::Free(buf); |
| } |
| |
| BLE_ERROR BLEEndPoint::Init(BleLayer * bleLayer, BLE_CONNECTION_OBJECT connObj, BleRole role, bool autoClose) |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| bool expectInitialAck; |
| |
| // Fail if already initialized. |
| VerifyOrExit(mBle == nullptr, err = BLE_ERROR_INCORRECT_STATE); |
| |
| // Validate args. |
| VerifyOrExit(bleLayer != nullptr, err = BLE_ERROR_BAD_ARGS); |
| VerifyOrExit(connObj != BLE_CONNECTION_UNINITIALIZED, err = BLE_ERROR_BAD_ARGS); |
| VerifyOrExit((role == kBleRole_Central || role == kBleRole_Peripheral), err = BLE_ERROR_BAD_ARGS); |
| |
| // Null-initialize callbacks and data members. |
| // |
| // Beware this line should we ever use virtuals in this class or its |
| // super(s). See similar lines in chip::System::Layer end points. |
| memset(this, 0, sizeof(*this)); |
| |
| // If end point plays peripheral role, expect ack for indication sent as last step of BTP handshake. |
| // If central, periperal's handshake indication 'ack's write sent by central to kick off the BTP handshake. |
| expectInitialAck = (role == kBleRole_Peripheral); |
| |
| err = mBtpEngine.Init(this, expectInitialAck); |
| if (err != BLE_NO_ERROR) |
| { |
| ChipLogError(Ble, "BtpEngine init failed"); |
| ExitNow(); |
| } |
| |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| err = (BLE_ERROR) mTxQueueMutex.Init(mTxQueueMutex); |
| if (err != BLE_NO_ERROR) |
| { |
| ChipLogError(Ble, "%s: Mutex init failed", __FUNCTION__); |
| ExitNow(); |
| } |
| err = mBtpEngineTest.Init(this); |
| if (err != BLE_NO_ERROR) |
| { |
| ChipLogError(Ble, "BTP test init failed"); |
| ExitNow(); |
| } |
| #endif |
| |
| // BleLayerObject initialization: |
| mBle = bleLayer; |
| mRefCount = 1; |
| |
| // BLEEndPoint data members: |
| mConnObj = connObj; |
| mRole = role; |
| mConnStateFlags = 0; |
| mTimerStateFlags = 0; |
| SetFlag(mConnStateFlags, kConnState_AutoClose, autoClose); |
| mLocalReceiveWindowSize = 0; |
| mRemoteReceiveWindowSize = 0; |
| mReceiveWindowMaxSize = 0; |
| mSendQueue = nullptr; |
| mAckToSend = nullptr; |
| |
| ChipLogDebugBleEndPoint(Ble, "initialized local rx window, size = %u", mLocalReceiveWindowSize); |
| |
| // End point is ready to connect or receive a connection. |
| mState = kState_Ready; |
| |
| exit: |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::SendCharacteristic(PacketBuffer * buf) |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| |
| if (mRole == kBleRole_Central) |
| { |
| if (!SendWrite(buf)) |
| { |
| err = BLE_ERROR_GATT_WRITE_FAILED; |
| } |
| else |
| { |
| // Write succeeded, so shrink remote receive window counter by 1. |
| mRemoteReceiveWindowSize -= 1; |
| ChipLogDebugBleEndPoint(Ble, "decremented remote rx window, new size = %u", mRemoteReceiveWindowSize); |
| } |
| } |
| else // (mRole == kBleRole_Peripheral), verified on Init |
| { |
| if (!SendIndication(buf)) |
| { |
| err = BLE_ERROR_GATT_INDICATE_FAILED; |
| } |
| else |
| { |
| // Indication succeeded, so shrink remote receive window counter by 1. |
| mRemoteReceiveWindowSize -= 1; |
| ChipLogDebugBleEndPoint(Ble, "decremented remote rx window, new size = %u", mRemoteReceiveWindowSize); |
| } |
| } |
| |
| return err; |
| } |
| |
| /* |
| * Routine to queue the Tx packet with a packet type |
| * kType_Data(0) - data packet |
| * kType_Control(1) - control packet |
| */ |
| void BLEEndPoint::QueueTx(PacketBuffer * data, PacketType_t type) |
| { |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| ChipLogDebugBleEndPoint(Ble, "%s: data->%p, type %d, len %d", __FUNCTION__, data, type, data->DataLength()); |
| mBtpEngine.PushPacketTag(data, type); |
| #endif |
| |
| QueueTxLock(); |
| |
| if (mSendQueue == nullptr) |
| { |
| mSendQueue = data; |
| ChipLogDebugBleEndPoint(Ble, "%s: Set data as new mSendQueue %p, type %d", __FUNCTION__, mSendQueue, type); |
| } |
| else |
| { |
| mSendQueue->AddToEnd(data); |
| ChipLogDebugBleEndPoint(Ble, "%s: Append data to mSendQueue %p, type %d", __FUNCTION__, mSendQueue, type); |
| } |
| |
| QueueTxUnlock(); |
| } |
| |
| BLE_ERROR BLEEndPoint::Send(PacketBuffer * data) |
| { |
| ChipLogDebugBleEndPoint(Ble, "entered Send"); |
| |
| BLE_ERROR err = BLE_NO_ERROR; |
| |
| VerifyOrExit(data != nullptr, err = BLE_ERROR_BAD_ARGS); |
| VerifyOrExit(IsConnected(mState), err = BLE_ERROR_INCORRECT_STATE); |
| |
| // Ensure outgoing message fits in a single contiguous PacketBuffer, as currently required by the |
| // message fragmentation and reassembly engine. |
| if (data->Next() != nullptr) |
| { |
| data->CompactHead(); |
| |
| if (data->Next() != nullptr) |
| { |
| err = BLE_ERROR_OUTBOUND_MESSAGE_TOO_BIG; |
| ExitNow(); |
| } |
| } |
| |
| // Add new message to send queue. |
| QueueTx(data, kType_Data); |
| data = nullptr; // Buffer freed when send queue freed on close, or on completion of current message transmission. |
| |
| // Send first fragment of new message, if we can. |
| err = DriveSending(); |
| SuccessOrExit(err); |
| |
| exit: |
| ChipLogDebugBleEndPoint(Ble, "exiting Send"); |
| |
| if (data != nullptr) |
| { |
| PacketBuffer::Free(data); |
| } |
| |
| if (err != BLE_NO_ERROR) |
| { |
| DoClose(kBleCloseFlag_AbortTransmission, err); |
| } |
| |
| return err; |
| } |
| |
| bool BLEEndPoint::PrepareNextFragment(PacketBuffer * data, bool & sentAck) |
| { |
| // If we have a pending fragment acknowledgement to send, piggyback it on the fragment we're about to transmit. |
| if (GetFlag(mTimerStateFlags, kTimerState_SendAckTimerRunning)) |
| { |
| // Reset local receive window counter. |
| mLocalReceiveWindowSize = mReceiveWindowMaxSize; |
| ChipLogDebugBleEndPoint(Ble, "reset local rx window on piggyback ack tx, size = %u", mLocalReceiveWindowSize); |
| |
| // Tell caller AND fragmenter we have an ack to piggyback. |
| sentAck = true; |
| } |
| else |
| { |
| // No ack to piggyback. |
| sentAck = false; |
| } |
| |
| return mBtpEngine.HandleCharacteristicSend(data, sentAck); |
| } |
| |
| BLE_ERROR BLEEndPoint::SendNextMessage() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| bool sentAck; |
| |
| // Get the first queued packet to send |
| QueueTxLock(); |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| // Return if tx queue is empty |
| // Note: DetachTail() does not check an empty queue |
| if (mSendQueue == NULL) |
| { |
| QueueTxUnlock(); |
| return err; |
| } |
| #endif |
| |
| PacketBuffer * data = mSendQueue; |
| mSendQueue = mSendQueue->DetachTail(); |
| QueueTxUnlock(); |
| |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| // Get and consume the packet tag in message buffer |
| PacketType_t type = mBtpEngine.PopPacketTag(data); |
| mBtpEngine.SetTxPacketType(type); |
| mBtpEngineTest.DoTxTiming(data, BTP_TX_START); |
| #endif |
| |
| // Hand whole message payload to the fragmenter. |
| VerifyOrExit(PrepareNextFragment(data, sentAck), err = BLE_ERROR_CHIPOBLE_PROTOCOL_ABORT); |
| data = nullptr; // Ownership passed to fragmenter's tx buf on PrepareNextFragment success. |
| |
| /* |
| // Todo: reenabled it after integrating fault injection |
| // Send first message fragment over the air. |
| CHIP_FAULT_INJECT(chip::FaultInjection::kFault_CHIPOBLESend, { |
| if (mRole == kBleRole_Central) |
| { |
| err = BLE_ERROR_GATT_WRITE_FAILED; |
| } |
| else |
| { |
| err = BLE_ERROR_GATT_INDICATE_FAILED; |
| } |
| ExitNow(); |
| }); |
| */ |
| err = SendCharacteristic(mBtpEngine.TxPacket()); |
| SuccessOrExit(err); |
| |
| if (sentAck) |
| { |
| // If sent piggybacked ack, stop send-ack timer. |
| StopSendAckTimer(); |
| } |
| |
| // Start ack received timer, if it's not already running. |
| err = StartAckReceivedTimer(); |
| SuccessOrExit(err); |
| |
| exit: |
| if (data != nullptr) |
| { |
| PacketBuffer::Free(data); |
| } |
| |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::ContinueMessageSend() |
| { |
| BLE_ERROR err; |
| bool sentAck; |
| |
| if (!PrepareNextFragment(nullptr, sentAck)) |
| { |
| // Log BTP error |
| ChipLogError(Ble, "btp fragmenter error on send!"); |
| mBtpEngine.LogState(); |
| |
| err = BLE_ERROR_CHIPOBLE_PROTOCOL_ABORT; |
| ExitNow(); |
| } |
| |
| err = SendCharacteristic(mBtpEngine.TxPacket()); |
| SuccessOrExit(err); |
| |
| if (sentAck) |
| { |
| // If sent piggybacked ack, stop send-ack timer. |
| StopSendAckTimer(); |
| } |
| |
| // Start ack received timer, if it's not already running. |
| err = StartAckReceivedTimer(); |
| SuccessOrExit(err); |
| |
| exit: |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::HandleHandshakeConfirmationReceived() |
| { |
| ChipLogDebugBleEndPoint(Ble, "entered HandleHandshakeConfirmationReceived"); |
| |
| BLE_ERROR err = BLE_NO_ERROR; |
| uint8_t closeFlags = kBleCloseFlag_AbortTransmission; |
| |
| // Free capabilities request/response payload. |
| QueueTxLock(); |
| mSendQueue = PacketBuffer::FreeHead(mSendQueue); |
| QueueTxUnlock(); |
| |
| if (mRole == kBleRole_Central) |
| { |
| // Subscribe to characteristic which peripheral will use to send indications. Prompts peripheral to send |
| // BLE transport capabilities indication. |
| VerifyOrExit(mBle->mPlatformDelegate->SubscribeCharacteristic(mConnObj, &CHIP_BLE_SVC_ID, &mBle->CHIP_BLE_CHAR_2_ID), |
| err = BLE_ERROR_GATT_SUBSCRIBE_FAILED); |
| |
| // We just sent a GATT subscribe request, so make sure to attempt unsubscribe on close. |
| SetFlag(mConnStateFlags, kConnState_DidBeginSubscribe, true); |
| |
| // Mark GATT operation in progress for subscribe request. |
| SetFlag(mConnStateFlags, kConnState_GattOperationInFlight, true); |
| } |
| else // (mRole == kBleRole_Peripheral), verified on Init |
| { |
| ChipLogDebugBleEndPoint(Ble, "got peripheral handshake indication confirmation"); |
| |
| if (mState == kState_Connected) // If we accepted BTP connection... |
| { |
| // If local receive window size has shrunk to or below immediate ack threshold, AND a message fragment is not |
| // pending on which to piggyback an ack, send immediate stand-alone ack. |
| if (mLocalReceiveWindowSize <= BLE_CONFIG_IMMEDIATE_ACK_WINDOW_THRESHOLD && mSendQueue == nullptr) |
| { |
| err = DriveStandAloneAck(); // Encode stand-alone ack and drive sending. |
| SuccessOrExit(err); |
| } |
| else |
| { |
| // Drive sending in case application callend Send() after we sent the handshake indication, but |
| // before the GATT confirmation for this indication was received. |
| err = DriveSending(); |
| SuccessOrExit(err); |
| } |
| } |
| else if (mState == kState_Aborting) // Else, if we rejected BTP connection... |
| { |
| closeFlags |= kBleCloseFlag_SuppressCallback; |
| err = BLE_ERROR_INCOMPATIBLE_PROTOCOL_VERSIONS; |
| ExitNow(); |
| } |
| } |
| |
| exit: |
| ChipLogDebugBleEndPoint(Ble, "exiting HandleHandshakeConfirmationReceived"); |
| |
| if (err != BLE_NO_ERROR) |
| { |
| DoClose(closeFlags, err); |
| } |
| |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::HandleFragmentConfirmationReceived() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| |
| ChipLogDebugBleEndPoint(Ble, "entered HandleFragmentConfirmationReceived"); |
| |
| // Suppress error logging if GATT confirmation overlaps with unsubscribe on final close. |
| if (IsUnsubscribePending()) |
| { |
| ChipLogDebugBleEndPoint(Ble, "send conf rx'd while unsubscribe in flight"); |
| ExitNow(); |
| } |
| |
| // Ensure we're in correct state to receive confirmation of non-handshake GATT send. |
| VerifyOrExit(IsConnected(mState), err = BLE_ERROR_INCORRECT_STATE); |
| |
| // TODO PacketBuffer high water mark optimization: if ack pending, but fragmenter state == complete, free fragmenter's |
| // tx buf before sending ack. |
| |
| if (GetFlag(mConnStateFlags, kConnState_StandAloneAckInFlight)) |
| { |
| // If confirmation was received for stand-alone ack, free its tx buffer. |
| PacketBuffer::Free(mAckToSend); |
| mAckToSend = nullptr; |
| |
| SetFlag(mConnStateFlags, kConnState_StandAloneAckInFlight, false); |
| } |
| |
| // If local receive window size has shrunk to or below immediate ack threshold, AND a message fragment is not |
| // pending on which to piggyback an ack, send immediate stand-alone ack. |
| // |
| // This check covers the case where the local receive window has shrunk between transmission and confirmation of |
| // the stand-alone ack, and also the case where a window size < the immediate ack threshold was detected in |
| // Receive(), but the stand-alone ack was deferred due to a pending outbound message fragment. |
| if (mLocalReceiveWindowSize <= BLE_CONFIG_IMMEDIATE_ACK_WINDOW_THRESHOLD && |
| !(mSendQueue != nullptr || mBtpEngine.TxState() == BtpEngine::kState_InProgress)) |
| { |
| err = DriveStandAloneAck(); // Encode stand-alone ack and drive sending. |
| SuccessOrExit(err); |
| } |
| else |
| { |
| err = DriveSending(); |
| SuccessOrExit(err); |
| } |
| |
| exit: |
| if (err != BLE_NO_ERROR) |
| { |
| DoClose(kBleCloseFlag_AbortTransmission, err); |
| } |
| |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::HandleGattSendConfirmationReceived() |
| { |
| ChipLogDebugBleEndPoint(Ble, "entered HandleGattSendConfirmationReceived"); |
| |
| // Mark outstanding GATT operation as finished. |
| SetFlag(mConnStateFlags, kConnState_GattOperationInFlight, false); |
| |
| // If confirmation was for outbound portion of BTP connect handshake... |
| if (!GetFlag(mConnStateFlags, kConnState_CapabilitiesConfReceived)) |
| { |
| SetFlag(mConnStateFlags, kConnState_CapabilitiesConfReceived, true); |
| |
| return HandleHandshakeConfirmationReceived(); |
| } |
| |
| return HandleFragmentConfirmationReceived(); |
| } |
| |
| BLE_ERROR BLEEndPoint::DriveStandAloneAck() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| |
| // Stop send-ack timer if running. |
| StopSendAckTimer(); |
| |
| // If stand-alone ack not already pending, allocate new payload buffer here. |
| if (mAckToSend == nullptr) |
| { |
| mAckToSend = PacketBuffer::New(); |
| VerifyOrExit(mAckToSend != nullptr, err = BLE_ERROR_NO_MEMORY); |
| } |
| |
| // Attempt to send stand-alone ack. |
| err = DriveSending(); |
| SuccessOrExit(err); |
| |
| exit: |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::DoSendStandAloneAck() |
| { |
| ChipLogDebugBleEndPoint(Ble, "entered DoSendStandAloneAck; sending stand-alone ack"); |
| |
| // Encode and transmit stand-alone ack. |
| mBtpEngine.EncodeStandAloneAck(mAckToSend); |
| BLE_ERROR err = SendCharacteristic(mAckToSend); |
| SuccessOrExit(err); |
| |
| // Reset local receive window counter. |
| mLocalReceiveWindowSize = mReceiveWindowMaxSize; |
| ChipLogDebugBleEndPoint(Ble, "reset local rx window on stand-alone ack tx, size = %u", mLocalReceiveWindowSize); |
| |
| SetFlag(mConnStateFlags, kConnState_StandAloneAckInFlight, true); |
| |
| // Start ack received timer, if it's not already running. |
| err = StartAckReceivedTimer(); |
| SuccessOrExit(err); |
| |
| exit: |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::DriveSending() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| |
| ChipLogDebugBleEndPoint(Ble, "entered DriveSending"); |
| |
| // If receiver's window is almost closed and we don't have an ack to send, OR we do have an ack to send but |
| // receiver's window is completely empty, OR another GATT operation is in flight, awaiting confirmation... |
| if ((mRemoteReceiveWindowSize <= BTP_WINDOW_NO_ACK_SEND_THRESHOLD && |
| !GetFlag(mTimerStateFlags, kTimerState_SendAckTimerRunning) && mAckToSend == nullptr) || |
| (mRemoteReceiveWindowSize == 0) || (GetFlag(mConnStateFlags, kConnState_GattOperationInFlight))) |
| { |
| #ifdef CHIP_BLE_END_POINT_DEBUG_LOGGING_ENABLED |
| if (mRemoteReceiveWindowSize <= BTP_WINDOW_NO_ACK_SEND_THRESHOLD && |
| !GetFlag(mTimerStateFlags, kTimerState_SendAckTimerRunning) && mAckToSend == NULL) |
| { |
| ChipLogDebugBleEndPoint(Ble, "NO SEND: receive window almost closed, and no ack to send"); |
| } |
| |
| if (mRemoteReceiveWindowSize == 0) |
| { |
| ChipLogDebugBleEndPoint(Ble, "NO SEND: remote receive window closed"); |
| } |
| |
| if (GetFlag(mConnStateFlags, kConnState_GattOperationInFlight)) |
| { |
| ChipLogDebugBleEndPoint(Ble, "NO SEND: Gatt op in flight"); |
| } |
| #endif |
| |
| // Can't send anything. |
| ExitNow(); |
| } |
| |
| // Otherwise, let's see what we can send. |
| |
| if (mAckToSend != nullptr) // If immediate, stand-alone ack is pending, send it. |
| { |
| err = DoSendStandAloneAck(); |
| SuccessOrExit(err); |
| } |
| else if (mBtpEngine.TxState() == BtpEngine::kState_Idle) // Else send next message fragment, if any. |
| { |
| // Fragmenter's idle, let's see what's in the send queue... |
| if (mSendQueue != nullptr) |
| { |
| // Transmit first fragment of next whole message in send queue. |
| err = SendNextMessage(); |
| SuccessOrExit(err); |
| } |
| else |
| { |
| // Nothing to send! |
| } |
| } |
| else if (mBtpEngine.TxState() == BtpEngine::kState_InProgress) |
| { |
| // Send next fragment of message currently held by fragmenter. |
| err = ContinueMessageSend(); |
| SuccessOrExit(err); |
| } |
| else if (mBtpEngine.TxState() == BtpEngine::kState_Complete) |
| { |
| // Clear fragmenter's pointer to sent message buffer and reset its Tx state. |
| PacketBuffer * sentBuf = mBtpEngine.TxPacket(); |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| mBtpEngineTest.DoTxTiming(sentBuf, BTP_TX_DONE); |
| #endif // CHIP_ENABLE_CHIPOBLE_TEST |
| mBtpEngine.ClearTxPacket(); |
| |
| // Free sent buffer. |
| PacketBuffer::Free(sentBuf); |
| sentBuf = nullptr; |
| |
| if (mSendQueue != nullptr) |
| { |
| // Transmit first fragment of next whole message in send queue. |
| err = SendNextMessage(); |
| SuccessOrExit(err); |
| } |
| else if (mState == kState_Closing && !mBtpEngine.ExpectingAck()) // and mSendQueue is NULL, per above... |
| { |
| // If end point closing, got last ack, and got out-of-order confirmation for last send, finalize close. |
| FinalizeClose(mState, kBleCloseFlag_SuppressCallback, BLE_NO_ERROR); |
| } |
| else |
| { |
| // Nothing to send! |
| } |
| } |
| |
| exit: |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::HandleCapabilitiesRequestReceived(PacketBuffer * data) |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| BleTransportCapabilitiesRequestMessage req; |
| BleTransportCapabilitiesResponseMessage resp; |
| PacketBuffer * responseBuf = nullptr; |
| uint16_t mtu; |
| |
| VerifyOrExit(data != nullptr, err = BLE_ERROR_BAD_ARGS); |
| |
| mState = kState_Connecting; |
| |
| // Decode BTP capabilities request. |
| err = BleTransportCapabilitiesRequestMessage::Decode((*data), req); |
| SuccessOrExit(err); |
| |
| responseBuf = PacketBuffer::New(); |
| VerifyOrExit(responseBuf != nullptr, err = BLE_ERROR_NO_MEMORY); |
| |
| // Determine BLE connection's negotiated ATT MTU, if possible. |
| if (req.mMtu > 0) // If MTU was observed and provided by central... |
| { |
| mtu = req.mMtu; // Accept central's observation of the MTU. |
| } |
| else // Otherwise, retrieve it via the platform delegate... |
| { |
| mtu = mBle->mPlatformDelegate->GetMTU(mConnObj); |
| } |
| |
| // Select fragment size for connection based on ATT MTU. |
| if (mtu > 0) // If one or both device knows connection's MTU... |
| { |
| resp.mFragmentSize = |
| chip::min(static_cast<uint16_t>(mtu - 3), BtpEngine::sMaxFragmentSize); // Reserve 3 bytes of MTU for ATT header. |
| } |
| else // Else, if neither device knows MTU... |
| { |
| ChipLogProgress(Ble, "cannot determine ATT MTU; selecting default fragment size = %u", BtpEngine::sDefaultFragmentSize); |
| resp.mFragmentSize = BtpEngine::sDefaultFragmentSize; |
| } |
| |
| // Select local and remote max receive window size based on local resources available for both incoming writes AND |
| // GATT confirmations. |
| mRemoteReceiveWindowSize = mLocalReceiveWindowSize = mReceiveWindowMaxSize = |
| chip::min(req.mWindowSize, static_cast<uint8_t>(BLE_MAX_RECEIVE_WINDOW_SIZE)); |
| resp.mWindowSize = mReceiveWindowMaxSize; |
| |
| ChipLogProgress(Ble, "local and remote recv window sizes = %u", resp.mWindowSize); |
| |
| // Select BLE transport protocol version from those supported by central, or none if no supported version found. |
| resp.mSelectedProtocolVersion = BleLayer::GetHighestSupportedProtocolVersion(req); |
| ChipLogProgress(Ble, "selected BTP version %d", resp.mSelectedProtocolVersion); |
| |
| if (resp.mSelectedProtocolVersion == kBleTransportProtocolVersion_None) |
| { |
| // If BLE transport protocol versions incompatible, prepare to close connection after subscription has been |
| // received and capabilities response has been sent. |
| ChipLogError(Ble, "incompatible BTP versions; peripheral expected between %d and %d", |
| CHIP_BLE_TRANSPORT_PROTOCOL_MIN_SUPPORTED_VERSION, CHIP_BLE_TRANSPORT_PROTOCOL_MAX_SUPPORTED_VERSION); |
| mState = kState_Aborting; |
| } |
| else if ((resp.mSelectedProtocolVersion == kBleTransportProtocolVersion_V1) || |
| (resp.mSelectedProtocolVersion == kBleTransportProtocolVersion_V2)) |
| { |
| // Set Rx and Tx fragment sizes to the same value |
| mBtpEngine.SetRxFragmentSize(resp.mFragmentSize); |
| mBtpEngine.SetTxFragmentSize(resp.mFragmentSize); |
| } |
| else // resp.SelectedProtocolVersion >= kBleTransportProtocolVersion_V3 |
| { |
| // This is the peripheral, so set Rx fragment size, and leave Tx at default |
| mBtpEngine.SetRxFragmentSize(resp.mFragmentSize); |
| } |
| ChipLogProgress(Ble, "using BTP fragment sizes rx %d / tx %d.", mBtpEngine.GetRxFragmentSize(), mBtpEngine.GetTxFragmentSize()); |
| |
| err = resp.Encode(responseBuf); |
| SuccessOrExit(err); |
| |
| // Stash capabilities response payload and wait for subscription from central. |
| QueueTx(responseBuf, kType_Data); |
| responseBuf = nullptr; |
| |
| // Start receive timer. Canceled when end point freed or connection established. |
| err = StartReceiveConnectionTimer(); |
| SuccessOrExit(err); |
| |
| exit: |
| if (responseBuf != nullptr) |
| { |
| PacketBuffer::Free(responseBuf); |
| } |
| |
| if (data != nullptr) |
| { |
| PacketBuffer::Free(data); |
| } |
| |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::HandleCapabilitiesResponseReceived(PacketBuffer * data) |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| BleTransportCapabilitiesResponseMessage resp; |
| |
| VerifyOrExit(data != nullptr, err = BLE_ERROR_BAD_ARGS); |
| |
| // Decode BTP capabilities response. |
| err = BleTransportCapabilitiesResponseMessage::Decode((*data), resp); |
| SuccessOrExit(err); |
| |
| VerifyOrExit(resp.mFragmentSize > 0, err = BLE_ERROR_INVALID_FRAGMENT_SIZE); |
| |
| ChipLogProgress(Ble, "peripheral chose BTP version %d; central expected between %d and %d", resp.mSelectedProtocolVersion, |
| CHIP_BLE_TRANSPORT_PROTOCOL_MIN_SUPPORTED_VERSION, CHIP_BLE_TRANSPORT_PROTOCOL_MAX_SUPPORTED_VERSION); |
| |
| if ((resp.mSelectedProtocolVersion < CHIP_BLE_TRANSPORT_PROTOCOL_MIN_SUPPORTED_VERSION) || |
| (resp.mSelectedProtocolVersion > CHIP_BLE_TRANSPORT_PROTOCOL_MAX_SUPPORTED_VERSION)) |
| { |
| err = BLE_ERROR_INCOMPATIBLE_PROTOCOL_VERSIONS; |
| ExitNow(); |
| } |
| |
| // Set fragment size as minimum of (reported ATT MTU, BTP characteristic size) |
| resp.mFragmentSize = chip::min(resp.mFragmentSize, BtpEngine::sMaxFragmentSize); |
| |
| if ((resp.mSelectedProtocolVersion == kBleTransportProtocolVersion_V1) || |
| (resp.mSelectedProtocolVersion == kBleTransportProtocolVersion_V2)) |
| { |
| mBtpEngine.SetRxFragmentSize(resp.mFragmentSize); |
| mBtpEngine.SetTxFragmentSize(resp.mFragmentSize); |
| } |
| else // resp.SelectedProtocolVersion >= kBleTransportProtocolVersion_V3 |
| { |
| // This is the central, so set Tx fragement size, and leave Rx at default. |
| mBtpEngine.SetTxFragmentSize(resp.mFragmentSize); |
| } |
| ChipLogProgress(Ble, "using BTP fragment sizes rx %d / tx %d.", mBtpEngine.GetRxFragmentSize(), mBtpEngine.GetTxFragmentSize()); |
| |
| // Select local and remote max receive window size based on local resources available for both incoming indications |
| // AND GATT confirmations. |
| mRemoteReceiveWindowSize = mLocalReceiveWindowSize = mReceiveWindowMaxSize = resp.mWindowSize; |
| |
| ChipLogProgress(Ble, "local and remote recv window size = %u", resp.mWindowSize); |
| |
| // Shrink local receive window counter by 1, since connect handshake indication requires acknowledgement. |
| mLocalReceiveWindowSize -= 1; |
| ChipLogDebugBleEndPoint(Ble, "decremented local rx window, new size = %u", mLocalReceiveWindowSize); |
| |
| // Send ack for connection handshake indication when timer expires. Sequence numbers always start at 0, |
| // and the reassembler's "last received seq num" is initialized to 0 and updated when new fragments are |
| // received from the peripheral, so we don't need to explicitly mark the ack num to send here. |
| err = StartSendAckTimer(); |
| SuccessOrExit(err); |
| |
| // We've sent a capabilities request write and received a compatible response, so the connect |
| // operation has completed successfully. |
| err = HandleConnectComplete(); |
| SuccessOrExit(err); |
| |
| exit: |
| if (data != nullptr) |
| { |
| PacketBuffer::Free(data); |
| } |
| |
| return err; |
| } |
| |
| // Returns number of open slots in remote receive window given the input values. |
| SequenceNumber_t BLEEndPoint::AdjustRemoteReceiveWindow(SequenceNumber_t lastReceivedAck, SequenceNumber_t maxRemoteWindowSize, |
| SequenceNumber_t newestUnackedSentSeqNum) |
| { |
| // Assumption: SequenceNumber_t is uint8_t. |
| // Assumption: Maximum possible sequence number value is UINT8_MAX. |
| // Assumption: Sequence numbers incremented past maximum value wrap to 0. |
| // Assumption: newest unacked sent sequence number never exceeds current (and by extension, new and un-wrapped) |
| // window boundary, so it never wraps relative to last received ack, if new window boundary would not |
| // also wrap. |
| |
| // Define new window boundary (inclusive) as uint16_t, so its value can temporarily exceed UINT8_MAX. |
| uint16_t newRemoteWindowBoundary = lastReceivedAck + maxRemoteWindowSize; |
| |
| if (newRemoteWindowBoundary > UINT8_MAX && newestUnackedSentSeqNum < lastReceivedAck) |
| { |
| // New window boundary WOULD wrap, and latest unacked seq num already HAS wrapped, so add offset to difference. |
| return (newRemoteWindowBoundary - (newestUnackedSentSeqNum + UINT8_MAX)); |
| } |
| |
| // Neither values would or have wrapped, OR new boundary WOULD wrap but latest unacked seq num does not, so no |
| // offset required. |
| return (newRemoteWindowBoundary - newestUnackedSentSeqNum); |
| } |
| |
| BLE_ERROR BLEEndPoint::Receive(PacketBuffer * data) |
| { |
| ChipLogDebugBleEndPoint(Ble, "+++++++++++++++++++++ entered receive"); |
| BLE_ERROR err = BLE_NO_ERROR; |
| SequenceNumber_t receivedAck = 0; |
| uint8_t closeFlags = kBleCloseFlag_AbortTransmission; |
| bool didReceiveAck = false; |
| |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| if (mBtpEngine.IsCommandPacket(data)) |
| { |
| ChipLogDebugBleEndPoint(Ble, "%s: Received Control frame: Flags %x", __FUNCTION__, *(data->Start())); |
| } |
| else |
| #endif |
| { // This is a special handling on the first CHIPoBLE data packet, the CapabilitiesRequest. |
| // Suppress error logging if peer's send overlaps with our unsubscribe on final close. |
| if (IsUnsubscribePending()) |
| { |
| ChipLogDebugBleEndPoint(Ble, "characteristic rx'd while unsubscribe in flight"); |
| ExitNow(); |
| } |
| |
| // If we're receiving the first inbound packet of a BLE transport connection handshake... |
| if (!GetFlag(mConnStateFlags, kConnState_CapabilitiesMsgReceived)) |
| { |
| if (mRole == kBleRole_Central) // If we're a central receiving a capabilities response indication... |
| { |
| // Ensure end point's in the right state before continuing. |
| VerifyOrExit(mState == kState_Connecting, err = BLE_ERROR_INCORRECT_STATE); |
| SetFlag(mConnStateFlags, kConnState_CapabilitiesMsgReceived, true); |
| |
| err = HandleCapabilitiesResponseReceived(data); |
| data = nullptr; |
| SuccessOrExit(err); |
| } |
| else // Or, a peripheral receiving a capabilities request write... |
| { |
| // Ensure end point's in the right state before continuing. |
| VerifyOrExit(mState == kState_Ready, err = BLE_ERROR_INCORRECT_STATE); |
| SetFlag(mConnStateFlags, kConnState_CapabilitiesMsgReceived, true); |
| |
| err = HandleCapabilitiesRequestReceived(data); |
| data = nullptr; |
| |
| if (err != BLE_NO_ERROR) |
| { |
| // If an error occurred decoding and handling the capabilities request, release the BLE connection. |
| // Central's connect attempt will time out if peripheral's application decides to keep the BLE |
| // connection open, or fail immediately if the application closes the connection. |
| closeFlags = closeFlags | kBleCloseFlag_SuppressCallback; |
| ExitNow(); |
| } |
| } |
| |
| // If received data was handshake packet, don't feed it to message reassembler. |
| ExitNow(); |
| } |
| } // End handling the CapabilitiesRequest |
| |
| ChipLogDebugBleEndPoint(Ble, "prepared to rx post-handshake btp packet"); |
| |
| // We've received a post-handshake BTP packet. |
| // Ensure end point's in the right state before continuing. |
| if (!IsConnected(mState)) |
| { |
| ChipLogError(Ble, "ep rx'd packet in bad state"); |
| err = BLE_ERROR_INCORRECT_STATE; |
| |
| ExitNow(); |
| } |
| |
| ChipLogDebugBleEndPoint(Ble, "BTP about to rx characteristic, state before:"); |
| mBtpEngine.LogStateDebug(); |
| |
| // Pass received packet into BTP protocol engine. |
| err = mBtpEngine.HandleCharacteristicReceived(data, receivedAck, didReceiveAck); |
| data = nullptr; // Buffer consumed by protocol engine; either freed or added to message reassembly area. |
| |
| ChipLogDebugBleEndPoint(Ble, "BTP rx'd characteristic, state after:"); |
| mBtpEngine.LogStateDebug(); |
| |
| SuccessOrExit(err); |
| |
| // Protocol engine accepted the fragment, so shrink local receive window counter by 1. |
| mLocalReceiveWindowSize -= 1; |
| ChipLogDebugBleEndPoint(Ble, "decremented local rx window, new size = %u", mLocalReceiveWindowSize); |
| |
| // Respond to received ack, if any. |
| if (didReceiveAck) |
| { |
| ChipLogDebugBleEndPoint(Ble, "got btp ack = %u", receivedAck); |
| |
| // If ack was rx'd for neweset unacked sent fragment, stop ack received timer. |
| if (!mBtpEngine.ExpectingAck()) |
| { |
| ChipLogDebugBleEndPoint(Ble, "got ack for last outstanding fragment"); |
| StopAckReceivedTimer(); |
| |
| if (mState == kState_Closing && mSendQueue == nullptr && mBtpEngine.TxState() == BtpEngine::kState_Idle) |
| { |
| // If end point closing, got confirmation for last send, and waiting for last ack, finalize close. |
| FinalizeClose(mState, kBleCloseFlag_SuppressCallback, BLE_NO_ERROR); |
| ExitNow(); |
| } |
| } |
| else // Else there are still sent fragments for which acks are expected, so restart ack received timer. |
| { |
| ChipLogDebugBleEndPoint(Ble, "still expecting ack(s), restarting timer..."); |
| err = RestartAckReceivedTimer(); |
| SuccessOrExit(err); |
| } |
| |
| ChipLogDebugBleEndPoint(Ble, "about to adjust remote rx window; got ack num = %u, newest unacked sent seq num = %u, \ |
| old window size = %u, max window size = %u", |
| receivedAck, mBtpEngine.GetNewestUnackedSentSequenceNumber(), mRemoteReceiveWindowSize, |
| mReceiveWindowMaxSize); |
| |
| // Open remote device's receive window according to sequence number it just acknowledged. |
| mRemoteReceiveWindowSize = |
| AdjustRemoteReceiveWindow(receivedAck, mReceiveWindowMaxSize, mBtpEngine.GetNewestUnackedSentSequenceNumber()); |
| |
| ChipLogDebugBleEndPoint(Ble, "adjusted remote rx window, new size = %u", mRemoteReceiveWindowSize); |
| |
| // Restart message transmission if it was previously paused due to window exhaustion. |
| err = DriveSending(); |
| SuccessOrExit(err); |
| } |
| |
| // The previous DriveSending() might have generated a piggyback acknowledgement if there was |
| // previously un-acked data. Otherwise, prepare to send acknowledgement for newly received fragment. |
| // |
| // If local receive window is below immediate ack threshold, AND there is no previous stand-alone ack in |
| // flight, AND there is no pending outbound message fragment on which the ack can and will be piggybacked, |
| // send immediate stand-alone ack to reopen window for sender. |
| // |
| // The "GATT operation in flight" check below covers "pending outbound message fragment" by extension, as when |
| // a message has been passed to the end point via Send(), its next outbound fragment must either be in flight |
| // itself, or awaiting the completion of another in-flight GATT operation. |
| // |
| // If any GATT operation is in flight that is NOT a stand-alone ack, the window size will be checked against |
| // this threshold again when the GATT operation is confirmed. |
| if (mBtpEngine.HasUnackedData()) |
| { |
| if (mLocalReceiveWindowSize <= BLE_CONFIG_IMMEDIATE_ACK_WINDOW_THRESHOLD && |
| !GetFlag(mConnStateFlags, kConnState_GattOperationInFlight)) |
| { |
| ChipLogDebugBleEndPoint(Ble, "sending immediate ack"); |
| err = DriveStandAloneAck(); |
| SuccessOrExit(err); |
| } |
| else |
| { |
| ChipLogDebugBleEndPoint(Ble, "starting send-ack timer"); |
| |
| // Send ack when timer expires. |
| err = StartSendAckTimer(); |
| SuccessOrExit(err); |
| } |
| } |
| |
| // If we've reassembled a whole message... |
| if (mBtpEngine.RxState() == BtpEngine::kState_Complete) |
| { |
| // Take ownership of message PacketBuffer |
| PacketBuffer * full_packet = mBtpEngine.RxPacket(); |
| mBtpEngine.ClearRxPacket(); |
| |
| ChipLogDebugBleEndPoint(Ble, "reassembled whole msg, len = %d", full_packet->DataLength()); |
| |
| #if CHIP_ENABLE_CHIPOBLE_TEST |
| // If we have a control message received callback, and end point is not closing... |
| if (mBtpEngine.RxPacketType() == kType_Control && OnCommandReceived && mState != kState_Closing) |
| { |
| ChipLogDebugBleEndPoint(Ble, "%s: calling OnCommandReceived, seq# %u, len = %u, type %u", __FUNCTION__, receivedAck, |
| full_packet->DataLength(), mBtpEngine.RxPacketType()); |
| // Pass received control message up the stack. |
| mBtpEngine.SetRxPacketSeq(receivedAck); |
| OnCommandReceived(this, full_packet); |
| } |
| else |
| #endif |
| // If we have a message received callback, and end point is not closing... |
| if (OnMessageReceived && mState != kState_Closing) |
| { |
| // Pass received message up the stack. |
| OnMessageReceived(this, full_packet); |
| } |
| else |
| { |
| // Free received message if there's no one to own it. |
| PacketBuffer::Free(full_packet); |
| } |
| } |
| |
| exit: |
| if (data != nullptr) |
| { |
| PacketBuffer::Free(data); |
| } |
| |
| if (err != BLE_NO_ERROR) |
| { |
| DoClose(closeFlags, err); |
| } |
| |
| return err; |
| } |
| |
| bool BLEEndPoint::SendWrite(PacketBuffer * buf) |
| { |
| // Add reference to message fragment for duration of platform's GATT write attempt. CHIP retains partial |
| // ownership of message fragment's PacketBuffer, since this is the same buffer as that of the whole message, just |
| // with a fragmenter-modified payload offset and data length. Buffer must be decref'd (i.e. PacketBuffer::Free'd) by |
| // platform when BLE GATT operation completes. |
| buf->AddRef(); |
| |
| SetFlag(mConnStateFlags, kConnState_GattOperationInFlight, true); |
| |
| return mBle->mPlatformDelegate->SendWriteRequest(mConnObj, &CHIP_BLE_SVC_ID, &mBle->CHIP_BLE_CHAR_1_ID, buf); |
| } |
| |
| bool BLEEndPoint::SendIndication(PacketBuffer * buf) |
| { |
| // Add reference to message fragment for duration of platform's GATT indication attempt. CHIP retains partial |
| // ownership of message fragment's PacketBuffer, since this is the same buffer as that of the whole message, just |
| // with a fragmenter-modified payload offset and data length. Buffer must be decref'd (i.e. PacketBuffer::Free'd) by |
| // platform when BLE GATT operation completes. |
| buf->AddRef(); |
| |
| SetFlag(mConnStateFlags, kConnState_GattOperationInFlight, true); |
| |
| return mBle->mPlatformDelegate->SendIndication(mConnObj, &CHIP_BLE_SVC_ID, &mBle->CHIP_BLE_CHAR_2_ID, buf); |
| } |
| |
| BLE_ERROR BLEEndPoint::StartConnectTimer() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| chip::System::Error timerErr; |
| |
| timerErr = mBle->mSystemLayer->StartTimer(BLE_CONNECT_TIMEOUT_MS, HandleConnectTimeout, this); |
| VerifyOrExit(timerErr == CHIP_SYSTEM_NO_ERROR, err = BLE_ERROR_START_TIMER_FAILED); |
| SetFlag(mTimerStateFlags, kTimerState_ConnectTimerRunning, true); |
| |
| exit: |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::StartReceiveConnectionTimer() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| chip::System::Error timerErr; |
| |
| timerErr = mBle->mSystemLayer->StartTimer(BLE_CONNECT_TIMEOUT_MS, HandleReceiveConnectionTimeout, this); |
| VerifyOrExit(timerErr == CHIP_SYSTEM_NO_ERROR, err = BLE_ERROR_START_TIMER_FAILED); |
| SetFlag(mTimerStateFlags, kTimerState_ReceiveConnectionTimerRunning, true); |
| |
| exit: |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::StartAckReceivedTimer() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| chip::System::Error timerErr; |
| |
| if (!GetFlag(mTimerStateFlags, kTimerState_AckReceivedTimerRunning)) |
| { |
| timerErr = mBle->mSystemLayer->StartTimer(BTP_ACK_RECEIVED_TIMEOUT_MS, HandleAckReceivedTimeout, this); |
| VerifyOrExit(timerErr == CHIP_SYSTEM_NO_ERROR, err = BLE_ERROR_START_TIMER_FAILED); |
| |
| SetFlag(mTimerStateFlags, kTimerState_AckReceivedTimerRunning, true); |
| } |
| |
| exit: |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::RestartAckReceivedTimer() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| |
| VerifyOrExit(GetFlag(mTimerStateFlags, kTimerState_AckReceivedTimerRunning), err = BLE_ERROR_INCORRECT_STATE); |
| |
| StopAckReceivedTimer(); |
| |
| err = StartAckReceivedTimer(); |
| SuccessOrExit(err); |
| |
| exit: |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::StartSendAckTimer() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| chip::System::Error timerErr; |
| |
| ChipLogDebugBleEndPoint(Ble, "entered StartSendAckTimer"); |
| |
| if (!GetFlag(mTimerStateFlags, kTimerState_SendAckTimerRunning)) |
| { |
| ChipLogDebugBleEndPoint(Ble, "starting new SendAckTimer"); |
| timerErr = mBle->mSystemLayer->StartTimer(BTP_ACK_SEND_TIMEOUT_MS, HandleSendAckTimeout, this); |
| VerifyOrExit(timerErr == CHIP_SYSTEM_NO_ERROR, err = BLE_ERROR_START_TIMER_FAILED); |
| |
| SetFlag(mTimerStateFlags, kTimerState_SendAckTimerRunning, true); |
| } |
| |
| exit: |
| return err; |
| } |
| |
| BLE_ERROR BLEEndPoint::StartUnsubscribeTimer() |
| { |
| BLE_ERROR err = BLE_NO_ERROR; |
| chip::System::Error timerErr; |
| |
| timerErr = mBle->mSystemLayer->StartTimer(BLE_UNSUBSCRIBE_TIMEOUT_MS, HandleUnsubscribeTimeout, this); |
| VerifyOrExit(timerErr == CHIP_SYSTEM_NO_ERROR, err = BLE_ERROR_START_TIMER_FAILED); |
| SetFlag(mTimerStateFlags, kTimerState_UnsubscribeTimerRunning, true); |
| |
| exit: |
| return err; |
| } |
| |
| void BLEEndPoint::StopConnectTimer() |
| { |
| // Cancel any existing connect timer. |
| mBle->mSystemLayer->CancelTimer(HandleConnectTimeout, this); |
| SetFlag(mTimerStateFlags, kTimerState_ConnectTimerRunning, false); |
| } |
| |
| void BLEEndPoint::StopReceiveConnectionTimer() |
| { |
| // Cancel any existing receive connection timer. |
| mBle->mSystemLayer->CancelTimer(HandleReceiveConnectionTimeout, this); |
| SetFlag(mTimerStateFlags, kTimerState_ReceiveConnectionTimerRunning, false); |
| } |
| |
| void BLEEndPoint::StopAckReceivedTimer() |
| { |
| // Cancel any existing ack-received timer. |
| mBle->mSystemLayer->CancelTimer(HandleAckReceivedTimeout, this); |
| SetFlag(mTimerStateFlags, kTimerState_AckReceivedTimerRunning, false); |
| } |
| |
| void BLEEndPoint::StopSendAckTimer() |
| { |
| // Cancel any existing send-ack timer. |
| mBle->mSystemLayer->CancelTimer(HandleSendAckTimeout, this); |
| SetFlag(mTimerStateFlags, kTimerState_SendAckTimerRunning, false); |
| } |
| |
| void BLEEndPoint::StopUnsubscribeTimer() |
| { |
| // Cancel any existing unsubscribe timer. |
| mBle->mSystemLayer->CancelTimer(HandleUnsubscribeTimeout, this); |
| SetFlag(mTimerStateFlags, kTimerState_UnsubscribeTimerRunning, false); |
| } |
| |
| void BLEEndPoint::HandleConnectTimeout(chip::System::Layer * systemLayer, void * appState, chip::System::Error err) |
| { |
| BLEEndPoint * ep = static_cast<BLEEndPoint *>(appState); |
| |
| // Check for event-based timer race condition. |
| if (GetFlag(ep->mTimerStateFlags, kTimerState_ConnectTimerRunning)) |
| { |
| ChipLogError(Ble, "connect handshake timed out, closing ep %p", ep); |
| SetFlag(ep->mTimerStateFlags, kTimerState_ConnectTimerRunning, false); |
| ep->DoClose(kBleCloseFlag_AbortTransmission, BLE_ERROR_CONNECT_TIMED_OUT); |
| } |
| } |
| |
| void BLEEndPoint::HandleReceiveConnectionTimeout(chip::System::Layer * systemLayer, void * appState, chip::System::Error err) |
| { |
| BLEEndPoint * ep = static_cast<BLEEndPoint *>(appState); |
| |
| // Check for event-based timer race condition. |
| if (GetFlag(ep->mTimerStateFlags, kTimerState_ReceiveConnectionTimerRunning)) |
| { |
| ChipLogError(Ble, "receive handshake timed out, closing ep %p", ep); |
| SetFlag(ep->mTimerStateFlags, kTimerState_ReceiveConnectionTimerRunning, false); |
| ep->DoClose(kBleCloseFlag_SuppressCallback | kBleCloseFlag_AbortTransmission, BLE_ERROR_RECEIVE_TIMED_OUT); |
| } |
| } |
| |
| void BLEEndPoint::HandleAckReceivedTimeout(chip::System::Layer * systemLayer, void * appState, chip::System::Error err) |
| { |
| BLEEndPoint * ep = static_cast<BLEEndPoint *>(appState); |
| |
| // Check for event-based timer race condition. |
| if (GetFlag(ep->mTimerStateFlags, kTimerState_AckReceivedTimerRunning)) |
| { |
| ChipLogError(Ble, "ack recv timeout, closing ep %p", ep); |
| ep->mBtpEngine.LogStateDebug(); |
| SetFlag(ep->mTimerStateFlags, kTimerState_AckReceivedTimerRunning, false); |
| ep->DoClose(kBleCloseFlag_AbortTransmission, BLE_ERROR_FRAGMENT_ACK_TIMED_OUT); |
| } |
| } |
| |
| void BLEEndPoint::HandleSendAckTimeout(chip::System::Layer * systemLayer, void * appState, chip::System::Error err) |
| { |
| BLEEndPoint * ep = static_cast<BLEEndPoint *>(appState); |
| |
| // Check for event-based timer race condition. |
| if (GetFlag(ep->mTimerStateFlags, kTimerState_SendAckTimerRunning)) |
| { |
| SetFlag(ep->mTimerStateFlags, kTimerState_SendAckTimerRunning, false); |
| |
| // If previous stand-alone ack isn't still in flight... |
| if (!GetFlag(ep->mConnStateFlags, kConnState_StandAloneAckInFlight)) |
| { |
| BLE_ERROR sendErr = ep->DriveStandAloneAck(); |
| |
| if (sendErr != BLE_NO_ERROR) |
| { |
| ep->DoClose(kBleCloseFlag_AbortTransmission, sendErr); |
| } |
| } |
| } |
| } |
| |
| void BLEEndPoint::HandleUnsubscribeTimeout(chip::System::Layer * systemLayer, void * appState, chip::System::Error err) |
| { |
| BLEEndPoint * ep = static_cast<BLEEndPoint *>(appState); |
| |
| // Check for event-based timer race condition. |
| if (GetFlag(ep->mTimerStateFlags, kTimerState_UnsubscribeTimerRunning)) |
| { |
| ChipLogError(Ble, "unsubscribe timed out, ble ep %p", ep); |
| SetFlag(ep->mTimerStateFlags, kTimerState_UnsubscribeTimerRunning, false); |
| ep->HandleUnsubscribeComplete(); |
| } |
| } |
| |
| } /* namespace Ble */ |
| } /* namespace chip */ |
| |
| #endif /* CONFIG_NETWORK_LAYER_BLE */ |