| /* |
| * |
| * 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 "TxtFields.h" |
| |
| #include <algorithm> |
| #include <cctype> |
| #include <climits> |
| #include <cstdio> |
| #include <inttypes.h> |
| #include <limits> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| #include <lib/dnssd/Advertiser.h> |
| #include <lib/dnssd/Resolver.h> |
| #include <lib/support/BytesToHex.h> |
| #include <lib/support/CHIPMemString.h> |
| #include <lib/support/SafeInt.h> |
| |
| namespace chip { |
| namespace Dnssd { |
| |
| namespace Internal { |
| |
| namespace { |
| |
| char SafeToLower(uint8_t ch) |
| { |
| return static_cast<char>(std::tolower(static_cast<unsigned char>(ch))); |
| } |
| bool IsKey(const ByteSpan & key, const char * desired) |
| { |
| if (key.size() != strlen(desired)) |
| { |
| return false; |
| } |
| for (size_t i = 0; i < key.size(); ++i) |
| { |
| if (SafeToLower(key.data()[i]) != SafeToLower(desired[i])) |
| { |
| return false; |
| } |
| } |
| return true; |
| } |
| |
| uint32_t MakeU32FromAsciiDecimal(const ByteSpan & val, uint32_t defaultValue = 0) |
| { |
| // +1 because `digits10` means the number of decimal digits that fit in `uint32_t`, |
| // not how many digits are enough to represent any `uint32_t` |
| // +1 for null-terminator |
| char nullTerminatedValue[std::numeric_limits<uint32_t>::digits10 + 2]; |
| |
| // value is too long to store `uint32_t` |
| if (val.size() >= sizeof(nullTerminatedValue)) |
| return defaultValue; |
| |
| // value contains leading zeros |
| if (val.size() > 1 && *val.data() == static_cast<uint8_t>('0')) |
| return defaultValue; |
| |
| Platform::CopyString(nullTerminatedValue, sizeof(nullTerminatedValue), val); |
| |
| char * endPtr; |
| unsigned long num = strtoul(nullTerminatedValue, &endPtr, 10); |
| |
| if (endPtr > nullTerminatedValue && *endPtr == '\0' && num != ULONG_MAX && CanCastTo<uint32_t>(num)) |
| return static_cast<uint32_t>(num); |
| |
| return defaultValue; |
| } |
| |
| uint16_t MakeU16FromAsciiDecimal(const ByteSpan & val) |
| { |
| const uint32_t num = MakeU32FromAsciiDecimal(val); |
| return CanCastTo<uint16_t>(num) ? static_cast<uint16_t>(num) : 0; |
| } |
| |
| uint8_t MakeU8FromAsciiDecimal(const ByteSpan & val) |
| { |
| const uint32_t num = MakeU32FromAsciiDecimal(val); |
| return CanCastTo<uint8_t>(num) ? static_cast<uint8_t>(num) : 0; |
| } |
| |
| bool MakeBoolFromAsciiDecimal(const ByteSpan & val) |
| { |
| return val.size() == 1 && static_cast<char>(*val.data()) == '1'; |
| } |
| |
| size_t GetPlusSignIdx(const ByteSpan & value) |
| { |
| // Fist value is the vendor id, second (after the +) is the product. |
| for (int i = 0; i < static_cast<int>(value.size()); ++i) |
| { |
| if (static_cast<char>(value.data()[i]) == '+') |
| { |
| return i; |
| } |
| } |
| return value.size(); |
| } |
| |
| } // namespace |
| |
| uint16_t GetProduct(const ByteSpan & value) |
| { |
| size_t plussign = GetPlusSignIdx(value); |
| if (plussign < value.size() - 1) |
| { |
| const uint8_t * productStrStart = value.data() + plussign + 1; |
| size_t productStrLen = value.size() - plussign - 1; |
| return MakeU16FromAsciiDecimal(ByteSpan(productStrStart, productStrLen)); |
| } |
| return 0; |
| } |
| |
| uint16_t GetVendor(const ByteSpan & value) |
| { |
| size_t plussign = GetPlusSignIdx(value); |
| return MakeU16FromAsciiDecimal(ByteSpan(value.data(), plussign)); |
| } |
| |
| uint16_t GetLongDiscriminator(const ByteSpan & value) |
| { |
| return MakeU16FromAsciiDecimal(value); |
| } |
| |
| uint8_t GetCommissioningMode(const ByteSpan & value) |
| { |
| return MakeU8FromAsciiDecimal(value); |
| } |
| |
| // TODO: possibly 32-bit? see spec issue #3226 |
| uint16_t GetDeviceType(const ByteSpan & value) |
| { |
| return MakeU16FromAsciiDecimal(value); |
| } |
| |
| void GetDeviceName(const ByteSpan & value, char * name) |
| { |
| Platform::CopyString(name, kMaxDeviceNameLen + 1, value); |
| } |
| |
| void GetRotatingDeviceId(const ByteSpan & value, uint8_t * rotatingId, size_t * len) |
| { |
| *len = Encoding::HexToBytes(reinterpret_cast<const char *>(value.data()), value.size(), rotatingId, kMaxRotatingIdLen); |
| } |
| |
| uint16_t GetPairingHint(const ByteSpan & value) |
| { |
| return MakeU16FromAsciiDecimal(value); |
| } |
| |
| void GetPairingInstruction(const ByteSpan & value, char * pairingInstruction) |
| { |
| Platform::CopyString(pairingInstruction, kMaxPairingInstructionLen + 1, value); |
| } |
| |
| Optional<System::Clock::Milliseconds32> GetRetryInterval(const ByteSpan & value) |
| { |
| const auto undefined = std::numeric_limits<uint32_t>::max(); |
| const auto retryInterval = MakeU32FromAsciiDecimal(value, undefined); |
| |
| if (retryInterval != undefined && retryInterval <= kMaxRetryInterval.count()) |
| return MakeOptional(System::Clock::Milliseconds32(retryInterval)); |
| |
| return NullOptional; |
| } |
| |
| TxtFieldKey GetTxtFieldKey(const ByteSpan & key) |
| { |
| for (auto & info : txtFieldInfo) |
| { |
| if (IsKey(key, info.keyStr)) |
| { |
| return info.key; |
| } |
| } |
| return TxtFieldKey::kUnknown; |
| } |
| |
| } // namespace Internal |
| |
| void FillNodeDataFromTxt(const ByteSpan & key, const ByteSpan & val, DiscoveredNodeData & nodeData) |
| { |
| TxtFieldKey keyType = Internal::GetTxtFieldKey(key); |
| switch (keyType) |
| { |
| case TxtFieldKey::kLongDiscriminator: |
| nodeData.longDiscriminator = Internal::GetLongDiscriminator(val); |
| break; |
| case TxtFieldKey::kVendorProduct: |
| nodeData.vendorId = Internal::GetVendor(val); |
| nodeData.productId = Internal::GetProduct(val); |
| break; |
| case TxtFieldKey::kCommissioningMode: |
| nodeData.commissioningMode = Internal::GetCommissioningMode(val); |
| break; |
| case TxtFieldKey::kDeviceType: |
| nodeData.deviceType = Internal::GetDeviceType(val); |
| break; |
| case TxtFieldKey::kDeviceName: |
| Internal::GetDeviceName(val, nodeData.deviceName); |
| break; |
| case TxtFieldKey::kRotatingDeviceId: |
| Internal::GetRotatingDeviceId(val, nodeData.rotatingId, &nodeData.rotatingIdLen); |
| break; |
| case TxtFieldKey::kPairingInstruction: |
| Internal::GetPairingInstruction(val, nodeData.pairingInstruction); |
| break; |
| case TxtFieldKey::kPairingHint: |
| nodeData.pairingHint = Internal::GetPairingHint(val); |
| break; |
| case TxtFieldKey::kMrpRetryIntervalIdle: |
| nodeData.mrpRetryIntervalIdle = Internal::GetRetryInterval(val); |
| break; |
| case TxtFieldKey::kMrpRetryIntervalActive: |
| nodeData.mrpRetryIntervalActive = Internal::GetRetryInterval(val); |
| break; |
| case TxtFieldKey::kTcpSupported: |
| nodeData.supportsTcp = Internal::MakeBoolFromAsciiDecimal(val); |
| break; |
| default: |
| break; |
| } |
| } |
| |
| void FillNodeDataFromTxt(const ByteSpan & key, const ByteSpan & value, ResolvedNodeData & nodeData) |
| { |
| switch (Internal::GetTxtFieldKey(key)) |
| { |
| case TxtFieldKey::kMrpRetryIntervalIdle: |
| nodeData.mMrpRetryIntervalIdle = Internal::GetRetryInterval(value); |
| break; |
| case TxtFieldKey::kMrpRetryIntervalActive: |
| nodeData.mMrpRetryIntervalActive = Internal::GetRetryInterval(value); |
| break; |
| case TxtFieldKey::kTcpSupported: |
| nodeData.mSupportsTcp = Internal::MakeBoolFromAsciiDecimal(value); |
| break; |
| default: |
| break; |
| } |
| } |
| |
| } // namespace Dnssd |
| } // namespace chip |