| /* |
| * |
| * Copyright (c) 2021 Project CHIP Authors |
| * |
| * 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. |
| */ |
| |
| #include "TransferFacilitator.h" |
| |
| #include <lib/core/CHIPError.h> |
| #include <lib/support/BitFlags.h> |
| #include <messaging/ExchangeContext.h> |
| #include <messaging/ExchangeDelegate.h> |
| #include <platform/CHIPDeviceLayer.h> |
| #include <protocols/bdx/BdxTransferSession.h> |
| #include <system/SystemClock.h> |
| #include <system/SystemLayer.h> |
| |
| namespace chip { |
| namespace bdx { |
| |
| constexpr System::Clock::Timeout TransferFacilitator::kDefaultPollFreq; |
| constexpr System::Clock::Timeout TransferFacilitator::kImmediatePollDelay; |
| |
| CHIP_ERROR TransferFacilitator::OnMessageReceived(chip::Messaging::ExchangeContext * ec, const chip::PayloadHeader & payloadHeader, |
| chip::System::PacketBufferHandle && payload) |
| { |
| if (mExchangeCtx == nullptr) |
| { |
| mExchangeCtx = ec; |
| } |
| |
| ChipLogDetail(BDX, "%s: message " ChipLogFormatMessageType " protocol " ChipLogFormatProtocolId, __FUNCTION__, |
| payloadHeader.GetMessageType(), ChipLogValueProtocolId(payloadHeader.GetProtocolID())); |
| CHIP_ERROR err = |
| mTransfer.HandleMessageReceived(payloadHeader, std::move(payload), System::SystemClock().GetMonotonicTimestamp()); |
| if (err != CHIP_NO_ERROR) |
| { |
| ChipLogError(BDX, "failed to handle message: %" CHIP_ERROR_FORMAT, err.Format()); |
| } |
| |
| // Almost every BDX message will follow up with a response on the exchange. Even messages that might signify the end of a |
| // transfer could necessitate a response if they are received at the wrong time. |
| // For this reason, it is left up to the application logic to call ExchangeContext::Close() when it has determined that the |
| // transfer is finished. |
| mExchangeCtx->WillSendMessage(); |
| |
| return err; |
| } |
| |
| void TransferFacilitator::OnResponseTimeout(Messaging::ExchangeContext * ec) |
| { |
| ChipLogError(BDX, "%s, ec: " ChipLogFormatExchange, __FUNCTION__, ChipLogValueExchange(ec)); |
| mExchangeCtx = nullptr; |
| mTransfer.Reset(); |
| } |
| |
| void TransferFacilitator::PollTimerHandler(chip::System::Layer * systemLayer, void * appState) |
| { |
| VerifyOrReturn(appState != nullptr); |
| static_cast<TransferFacilitator *>(appState)->PollForOutput(); |
| } |
| |
| void TransferFacilitator::PollForOutput() |
| { |
| TransferSession::OutputEvent outEvent; |
| mTransfer.PollOutput(outEvent, System::SystemClock().GetMonotonicTimestamp()); |
| HandleTransferSessionOutput(outEvent); |
| |
| VerifyOrReturn(mSystemLayer != nullptr, ChipLogError(BDX, "%s mSystemLayer is null", __FUNCTION__)); |
| if (!mStopPolling) |
| { |
| mSystemLayer->StartTimer(mPollFreq, PollTimerHandler, this); |
| } |
| else |
| { |
| mSystemLayer->CancelTimer(PollTimerHandler, this); |
| mStopPolling = false; |
| } |
| } |
| |
| void TransferFacilitator::ScheduleImmediatePoll() |
| { |
| VerifyOrReturn(mSystemLayer != nullptr, ChipLogError(BDX, "%s mSystemLayer is null", __FUNCTION__)); |
| mSystemLayer->StartTimer(System::Clock::Milliseconds32(kImmediatePollDelay), PollTimerHandler, this); |
| } |
| |
| CHIP_ERROR Responder::PrepareForTransfer(System::Layer * layer, TransferRole role, BitFlags<TransferControlFlags> xferControlOpts, |
| uint16_t maxBlockSize, System::Clock::Timeout timeout, System::Clock::Timeout pollFreq) |
| { |
| VerifyOrReturnError(layer != nullptr, CHIP_ERROR_INVALID_ARGUMENT); |
| |
| mPollFreq = pollFreq; |
| mSystemLayer = layer; |
| |
| ReturnErrorOnFailure(mTransfer.WaitForTransfer(role, xferControlOpts, maxBlockSize, timeout)); |
| |
| ChipLogProgress(BDX, "Start polling for messages"); |
| mStopPolling = false; |
| mSystemLayer->StartTimer(mPollFreq, PollTimerHandler, this); |
| return CHIP_NO_ERROR; |
| } |
| |
| void Responder::ResetTransfer() |
| { |
| mTransfer.Reset(); |
| ChipLogProgress(BDX, "Stop polling for messages"); |
| mStopPolling = true; |
| } |
| |
| CHIP_ERROR Initiator::InitiateTransfer(System::Layer * layer, TransferRole role, const TransferSession::TransferInitData & initData, |
| System::Clock::Timeout timeout, System::Clock::Timeout pollFreq) |
| { |
| VerifyOrReturnError(layer != nullptr, CHIP_ERROR_INVALID_ARGUMENT); |
| |
| mPollFreq = pollFreq; |
| mSystemLayer = layer; |
| |
| ReturnErrorOnFailure(mTransfer.StartTransfer(role, initData, timeout)); |
| |
| mSystemLayer->StartTimer(mPollFreq, PollTimerHandler, this); |
| return CHIP_NO_ERROR; |
| } |
| |
| } // namespace bdx |
| } // namespace chip |