| /* |
| * |
| * Copyright (c) 2022 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 <lib/dnssd/IPAddressSorter.h> |
| #include <lib/support/SortUtils.h> |
| |
| namespace chip { |
| namespace Dnssd { |
| namespace IPAddressSorter { |
| |
| void Sort(Inet::IPAddress * addresses, size_t count, Inet::InterfaceId interfaceId) |
| { |
| Sorting::BubbleSort(addresses, count, [interfaceId](const Inet::IPAddress & a, const Inet::IPAddress & b) -> bool { |
| auto scoreA = to_underlying(ScoreIpAddress(a, interfaceId)); |
| auto scoreB = to_underlying(ScoreIpAddress(b, interfaceId)); |
| return scoreA > scoreB; |
| }); |
| } |
| |
| void Sort(const Span<Inet::IPAddress> & addresses, Inet::InterfaceId interfaceId) |
| { |
| Sorting::BubbleSort(addresses.begin(), addresses.size(), |
| [interfaceId](const Inet::IPAddress & a, const Inet::IPAddress & b) -> bool { |
| auto scoreA = to_underlying(ScoreIpAddress(a, interfaceId)); |
| auto scoreB = to_underlying(ScoreIpAddress(b, interfaceId)); |
| return scoreA > scoreB; |
| }); |
| } |
| |
| IpScore ScoreIpAddress(const Inet::IPAddress & ip, Inet::InterfaceId interfaceId) |
| { |
| if (ip.IsIPv6()) |
| { |
| #ifdef __APPLE__ |
| if (ip.IsIPv6LinkLocal()) |
| { |
| return IpScore::kLinkLocal; |
| } |
| #endif // __APPLE__ |
| |
| if (interfaceId.MatchLocalIPv6Subnet(ip)) |
| { |
| if (ip.IsIPv6GlobalUnicast()) |
| { |
| return IpScore::kGlobalUnicastWithSharedPrefix; |
| } |
| if (ip.IsIPv6ULA()) |
| { |
| return IpScore::kUniqueLocalWithSharedPrefix; |
| } |
| } |
| if (ip.IsIPv6GlobalUnicast()) |
| { |
| return IpScore::kGlobalUnicast; |
| } |
| |
| if (ip.IsIPv6ULA()) |
| { |
| return IpScore::kUniqueLocal; |
| } |
| |
| #ifndef __APPLE__ |
| if (ip.IsIPv6LinkLocal()) |
| { |
| return IpScore::kLinkLocal; |
| } |
| #endif // __APPLE__ |
| |
| return IpScore::kOtherIpv6; |
| } |
| |
| return IpScore::kIpv4; |
| } |
| |
| } // namespace IPAddressSorter |
| } // namespace Dnssd |
| } // namespace chip |