blob: ee546d4db449d52e733e5b943c5cbe89a08b8cf6 [file] [log] [blame]
/*
*
* Copyright (c) 2020 Project CHIP Authors
* All rights reserved.
*
* 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 an object for a CHIP Echo unsolicited
* initiator (client).
*
*/
#include "Echo.h"
namespace chip {
namespace Protocols {
namespace Echo {
// The Echo message timeout value in milliseconds.
constexpr System::Clock::Timeout kEchoMessageTimeout = System::Clock::Milliseconds32(800);
CHIP_ERROR EchoClient::Init(Messaging::ExchangeManager * exchangeMgr, const SessionHandle & session)
{
// Error if already initialized.
if (mExchangeMgr != nullptr)
return CHIP_ERROR_INCORRECT_STATE;
mExchangeMgr = exchangeMgr;
mSecureSession.Grab(session);
OnEchoResponseReceived = nullptr;
mExchangeCtx = nullptr;
return CHIP_NO_ERROR;
}
void EchoClient::Shutdown()
{
if (mExchangeCtx != nullptr)
{
mExchangeCtx->Abort();
mExchangeCtx = nullptr;
}
OnEchoResponseReceived = nullptr;
mExchangeMgr = nullptr;
}
CHIP_ERROR EchoClient::SendEchoRequest(System::PacketBufferHandle && payload, Messaging::SendFlags sendFlags)
{
CHIP_ERROR err = CHIP_NO_ERROR;
// Discard any existing exchange context. Effectively we can only have one Echo exchange with
// a single node at any one time.
if (mExchangeCtx != nullptr)
{
mExchangeCtx->Abort();
mExchangeCtx = nullptr;
}
VerifyOrReturnError(mSecureSession, CHIP_ERROR_INVALID_MESSAGE_TYPE);
// Create a new exchange context.
mExchangeCtx = mExchangeMgr->NewContext(mSecureSession.Get().Value(), this);
if (mExchangeCtx == nullptr)
{
return CHIP_ERROR_NO_MEMORY;
}
mExchangeCtx->SetResponseTimeout(kEchoMessageTimeout);
// Send an Echo Request message. Discard the exchange context if the send fails.
err = mExchangeCtx->SendMessage(MsgType::EchoRequest, std::move(payload),
sendFlags.Set(Messaging::SendMessageFlags::kExpectResponse));
if (err != CHIP_NO_ERROR)
{
mExchangeCtx->Abort();
mExchangeCtx = nullptr;
}
return err;
}
CHIP_ERROR EchoClient::OnMessageReceived(Messaging::ExchangeContext * ec, const PayloadHeader & payloadHeader,
System::PacketBufferHandle && payload)
{
// Assert that the exchange context matches the client's current context.
// This should never fail because even if SendEchoRequest is called
// back-to-back, the second call will call Close() on the first exchange,
// which clears the OnMessageReceived callback.
VerifyOrDie(ec == mExchangeCtx);
mExchangeCtx = nullptr;
// Verify that the message is an Echo Response.
if (!payloadHeader.HasMessageType(MsgType::EchoResponse))
{
return CHIP_ERROR_INVALID_ARGUMENT;
}
// Call the registered OnEchoResponseReceived handler, if any.
if (OnEchoResponseReceived != nullptr)
{
OnEchoResponseReceived(ec, std::move(payload));
}
return CHIP_NO_ERROR;
}
void EchoClient::OnResponseTimeout(Messaging::ExchangeContext * ec)
{
mExchangeCtx = nullptr;
ChipLogProgress(Echo, "Time out! failed to receive echo response from Exchange: %p", ec);
}
} // namespace Echo
} // namespace Protocols
} // namespace chip