blob: 9d59e2dfdbca03c8c76ff4b480fa57acfc390d0a [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.
*/
// module header, comes first
#include <controller/AbstractDnssdDiscoveryController.h>
#include <lib/core/CHIPEncoding.h>
#include <lib/support/logging/CHIPLogging.h>
#include <bitset>
namespace chip {
namespace Controller {
static bool SameExceptOrder(const chip::Span<const Inet::IPAddress> & v1, const chip::Span<const Inet::IPAddress> & v2)
{
std::bitset<chip::Dnssd::CommonResolutionData::kMaxIPAddresses> addressUsed;
VerifyOrDie(v1.size() <= Dnssd::CommissionNodeData::kMaxIPAddresses && v2.size() <= Dnssd::CommissionNodeData::kMaxIPAddresses);
if (v1.size() != v2.size())
{
return false;
}
for (size_t s = 0; s < v1.size(); s++)
{
for (size_t d = 0; d < v2.size(); d++)
{
if (!addressUsed[d] && v1[s] == v2[d])
{
// Change the used flag so that the compared target is no longer used
addressUsed.set(d, true);
break;
}
}
}
return addressUsed.count() == v2.size();
}
void AbstractDnssdDiscoveryController::OnNodeDiscovered(const chip::Dnssd::DiscoveredNodeData & discNodeData)
{
VerifyOrReturn(discNodeData.Is<chip::Dnssd::CommissionNodeData>());
auto discoveredNodes = GetDiscoveredNodes();
auto & nodeData = discNodeData.Get<chip::Dnssd::CommissionNodeData>();
for (auto & discoveredNode : discoveredNodes)
{
if (!discoveredNode.IsValid())
{
continue;
}
chip::Span<const Inet::IPAddress> discoveredNodeIPAddressSpan(&discoveredNode.ipAddress[0], discoveredNode.numIPs);
chip::Span<const Inet::IPAddress> nodeDataIPAddressSpan(&nodeData.ipAddress[0], nodeData.numIPs);
if (strcmp(discoveredNode.hostName, nodeData.hostName) == 0 && discoveredNode.port == nodeData.port &&
SameExceptOrder(discoveredNodeIPAddressSpan, nodeDataIPAddressSpan))
{
discoveredNode = nodeData;
if (mDeviceDiscoveryDelegate != nullptr)
{
mDeviceDiscoveryDelegate->OnDiscoveredDevice(nodeData);
}
return;
}
}
// Node not yet in the list
for (auto & discoveredNode : discoveredNodes)
{
if (!discoveredNode.IsValid())
{
discoveredNode = nodeData;
if (mDeviceDiscoveryDelegate != nullptr)
{
mDeviceDiscoveryDelegate->OnDiscoveredDevice(nodeData);
}
return;
}
}
ChipLogError(Discovery, "Failed to add discovered node with hostname %s- Insufficient space", nodeData.hostName);
}
CHIP_ERROR AbstractDnssdDiscoveryController::SetUpNodeDiscovery()
{
auto discoveredNodes = GetDiscoveredNodes();
for (auto & discoveredNode : discoveredNodes)
{
discoveredNode.Reset();
}
return CHIP_NO_ERROR;
}
const Dnssd::CommissionNodeData * AbstractDnssdDiscoveryController::GetDiscoveredNode(int idx)
{
// TODO(cecille): Add assertion about main loop.
auto discoveredNodes = GetDiscoveredNodes();
if (0 <= idx && idx < CHIP_DEVICE_CONFIG_MAX_DISCOVERED_NODES && discoveredNodes.data()[idx].IsValid())
{
return discoveredNodes.data() + idx;
}
return nullptr;
}
} // namespace Controller
} // namespace chip