-
Notifications
You must be signed in to change notification settings - Fork 2.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Implemented CASEServer to fetch credentials, and wait for SigmaR1 mes…
…sage
- Loading branch information
Showing
6 changed files
with
228 additions
and
15 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,119 @@ | ||
/* | ||
* | ||
* 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 <protocols/secure_channel/CASEServer.h> | ||
|
||
#include <core/CHIPError.h> | ||
#include <support/CodeUtils.h> | ||
#include <support/SafeInt.h> | ||
#include <transport/SecureSessionMgr.h> | ||
|
||
using namespace ::chip::Inet; | ||
using namespace ::chip::Transport; | ||
|
||
namespace chip { | ||
|
||
CHIP_ERROR CASEServer::WaitForSessionEstablishment(Messaging::ExchangeManager * exchangeManager, TransportMgrBase * transportMgr, | ||
SecureSessionMgr * sessionMgr, Transport::AdminPairingTable * admins) | ||
{ | ||
VerifyOrReturnError(transportMgr != nullptr, CHIP_ERROR_INVALID_ARGUMENT); | ||
VerifyOrReturnError(exchangeManager != nullptr, CHIP_ERROR_INVALID_ARGUMENT); | ||
VerifyOrReturnError(sessionMgr != nullptr, CHIP_ERROR_INVALID_ARGUMENT); | ||
VerifyOrReturnError(admins != nullptr, CHIP_ERROR_INVALID_ARGUMENT); | ||
|
||
mSessionMgr = sessionMgr; | ||
mAdmins = admins; | ||
mExchangeManager = exchangeManager; | ||
|
||
ReturnErrorOnFailure(mPairingSession.MessageDispatch().Init(transportMgr)); | ||
|
||
ReturnErrorOnFailure( | ||
mExchangeManager->RegisterUnsolicitedMessageHandlerForType(Protocols::SecureChannel::MsgType::CASE_SigmaR1, this)); | ||
return CHIP_NO_ERROR; | ||
} | ||
|
||
CHIP_ERROR CASEServer::InitCASEHandshake(Messaging::ExchangeContext * ec) | ||
{ | ||
ReturnErrorCodeIf(ec == nullptr, CHIP_ERROR_INVALID_ARGUMENT); | ||
|
||
// Lookup the admin that corresponds to the CASE session setup request. | ||
// Each admin provisions their own credentials on the device. So it's essential to | ||
// use the correct operational certificates for CASE session setup. | ||
mAdminId = ec->GetSecureSession().GetAdminId(); | ||
ReturnErrorCodeIf(mAdminId == Transport::kUndefinedAdminId, CHIP_ERROR_INVALID_ARGUMENT); | ||
|
||
Transport::AdminPairingInfo * admin = mAdmins->FindAdminWithId(mAdminId); | ||
ReturnErrorCodeIf(admin == nullptr, CHIP_ERROR_INVALID_ARGUMENT); | ||
|
||
ReturnErrorOnFailure(admin->GetOperationalCertificateSet(mCertificates)); | ||
|
||
mCredentials.Release(); | ||
ReturnErrorOnFailure(mCredentials.Init(&mCertificates, mCertificates.GetCertCount())); | ||
|
||
// Setup CASE state machine using the credentials for the current admin. | ||
ReturnErrorOnFailure(mPairingSession.WaitForSessionEstablishment(&mCredentials, mNextKeyId++, this)); | ||
|
||
// Hand over the exchange context to the CASE session. | ||
ec->SetDelegate(&mPairingSession); | ||
|
||
return CHIP_NO_ERROR; | ||
} | ||
|
||
void CASEServer::OnMessageReceived(Messaging::ExchangeContext * ec, const PacketHeader & packetHeader, | ||
const PayloadHeader & payloadHeader, System::PacketBufferHandle payload) | ||
{ | ||
ReturnOnFailure(InitCASEHandshake(ec)); | ||
mPairingSession.OnMessageReceived(ec, packetHeader, payloadHeader, std::move(payload)); | ||
|
||
// TODO - Enable multiple concurrent CASE session establishment | ||
// This will prevent CASEServer to process another CASE session establishment request until the current | ||
// one completes (successfully or failed) | ||
mExchangeManager->UnregisterUnsolicitedMessageHandlerForType(Protocols::SecureChannel::MsgType::CASE_SigmaR1); | ||
} | ||
|
||
void CASEServer::Cleanup() | ||
{ | ||
// Let's re-register for CASE SigmaR1 message, so that the next CASE session setup request can be processed. | ||
mExchangeManager->RegisterUnsolicitedMessageHandlerForType(Protocols::SecureChannel::MsgType::CASE_SigmaR1, this); | ||
mAdminId = Transport::kUndefinedAdminId; | ||
mCredentials.Release(); | ||
} | ||
|
||
void CASEServer::OnSessionEstablishmentError(CHIP_ERROR err) | ||
{ | ||
ChipLogProgress(AppServer, "CASE Session establishment failed: %s", ErrorStr(err)); | ||
Cleanup(); | ||
} | ||
|
||
void CASEServer::OnSessionEstablished() | ||
{ | ||
ChipLogProgress(AppServer, "CASE Session established. Setting up the secure channel."); | ||
CHIP_ERROR err = | ||
mSessionMgr->NewPairing(Optional<Transport::PeerAddress>::Value(mPairingSession.PeerConnection().GetPeerAddress()), | ||
mPairingSession.PeerConnection().GetPeerNodeId(), &mPairingSession, | ||
SecureSession::SessionRole::kResponder, mAdminId, nullptr); | ||
if (err != CHIP_NO_ERROR) | ||
{ | ||
ChipLogError(Ble, "Failed in setting up secure channel: err %s", ErrorStr(err)); | ||
OnSessionEstablishmentError(err); | ||
return; | ||
} | ||
|
||
ChipLogProgress(AppServer, "CASE secure channel is available now."); | ||
Cleanup(); | ||
} | ||
} // namespace chip |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,78 @@ | ||
/* | ||
* | ||
* 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. | ||
*/ | ||
|
||
#pragma once | ||
|
||
#include <messaging/ExchangeDelegate.h> | ||
#include <messaging/ExchangeMgr.h> | ||
#include <protocols/secure_channel/CASESession.h> | ||
|
||
namespace chip { | ||
|
||
class CASEServer : public SessionEstablishmentDelegate, public Messaging::ExchangeDelegateBase | ||
{ | ||
public: | ||
CASEServer() {} | ||
~CASEServer() | ||
{ | ||
if (mExchangeManager != nullptr) | ||
{ | ||
mExchangeManager->UnregisterUnsolicitedMessageHandlerForType(Protocols::SecureChannel::MsgType::CASE_SigmaR1); | ||
} | ||
|
||
mCredentials.Release(); | ||
} | ||
|
||
CHIP_ERROR WaitForSessionEstablishment(Messaging::ExchangeManager * exchangeManager, TransportMgrBase * transportMgr, | ||
SecureSessionMgr * sessionMgr, Transport::AdminPairingTable * admins); | ||
|
||
//////////// SessionEstablishmentDelegate Implementation /////////////// | ||
void OnSessionEstablishmentError(CHIP_ERROR error) override; | ||
void OnSessionEstablished() override; | ||
|
||
void Cleanup(); | ||
|
||
uint16_t GetNextKeyId() const { return mNextKeyId; } | ||
void SetNextKeyId(uint16_t id) { mNextKeyId = id; } | ||
|
||
//// ExchangeDelegate Implementation //// | ||
void OnMessageReceived(Messaging::ExchangeContext * ec, const PacketHeader & packetHeader, const PayloadHeader & payloadHeader, | ||
System::PacketBufferHandle payload) override; | ||
void OnResponseTimeout(Messaging::ExchangeContext * ec) override {} | ||
Messaging::ExchangeMessageDispatch * GetMessageDispatch(Messaging::ReliableMessageMgr * reliableMessageManager, | ||
SecureSessionMgr * sessionMgr) override | ||
{ | ||
return mPairingSession.GetMessageDispatch(reliableMessageManager, sessionMgr); | ||
} | ||
|
||
private: | ||
Messaging::ExchangeManager * mExchangeManager = nullptr; | ||
|
||
CASESession mPairingSession; | ||
uint16_t mNextKeyId = 0; | ||
SecureSessionMgr * mSessionMgr = nullptr; | ||
|
||
Transport::AdminId mAdminId = Transport::kUndefinedAdminId; | ||
|
||
Transport::AdminPairingTable * mAdmins = nullptr; | ||
ChipCertificateSet mCertificates; | ||
OperationalCredentialSet mCredentials; | ||
|
||
CHIP_ERROR InitCASEHandshake(Messaging::ExchangeContext * ec); | ||
}; | ||
|
||
} // namespace chip |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters