/* * * 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 #include #include #include namespace { using namespace chip; class TestThreadOperationalDataset : public ::testing::Test { public: static void SetUpTestSuite() { ASSERT_EQ(chip::Platform::MemoryInit(), CHIP_NO_ERROR); } static void TearDownTestSuite() { chip::Platform::MemoryShutdown(); } Thread::OperationalDataset dataset; }; TEST_F(TestThreadOperationalDataset, TestInit) { uint8_t longerThanOperationalDatasetSize[255]{}; EXPECT_EQ(dataset.Init(ByteSpan(longerThanOperationalDatasetSize)), CHIP_ERROR_INVALID_ARGUMENT); EXPECT_EQ(dataset.Init(ByteSpan()), CHIP_NO_ERROR); { uint8_t data[] = { 0x01, 0x02, 0x03 }; EXPECT_EQ(dataset.Init(ByteSpan(data)), CHIP_ERROR_INVALID_ARGUMENT); } { uint8_t data[] = { 0x01 }; EXPECT_EQ(dataset.Init(ByteSpan(data)), CHIP_ERROR_INVALID_ARGUMENT); } } TEST_F(TestThreadOperationalDataset, TestActiveTimestamp) { static constexpr uint64_t kActiveTimestampValue = 1; uint64_t activeTimestamp = 0; EXPECT_EQ(dataset.SetActiveTimestamp(kActiveTimestampValue), CHIP_NO_ERROR); EXPECT_EQ(dataset.GetActiveTimestamp(activeTimestamp), CHIP_NO_ERROR); EXPECT_EQ(activeTimestamp, kActiveTimestampValue); } TEST_F(TestThreadOperationalDataset, TestChannel) { static constexpr uint16_t kChannelValue = 15; uint16_t channel = 0; EXPECT_EQ(dataset.SetChannel(kChannelValue), CHIP_NO_ERROR); EXPECT_EQ(dataset.GetChannel(channel), CHIP_NO_ERROR); EXPECT_EQ(channel, kChannelValue); } TEST_F(TestThreadOperationalDataset, TestExtendedPanId) { static constexpr uint8_t kExtendedPanId[] = { 0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77 }; uint8_t extendedPanId[Thread::kSizeExtendedPanId] = { 0 }; EXPECT_EQ(dataset.SetExtendedPanId(kExtendedPanId), CHIP_NO_ERROR); EXPECT_EQ(dataset.GetExtendedPanId(extendedPanId), CHIP_NO_ERROR); EXPECT_EQ(memcmp(extendedPanId, kExtendedPanId, sizeof(kExtendedPanId)), 0); ByteSpan span; EXPECT_EQ(dataset.GetExtendedPanIdAsByteSpan(span), CHIP_NO_ERROR); EXPECT_EQ(span.size(), sizeof(kExtendedPanId)); EXPECT_EQ(memcmp(extendedPanId, span.data(), sizeof(kExtendedPanId)), 0); } TEST_F(TestThreadOperationalDataset, TestMasterKey) { static constexpr uint8_t kMasterKey[] = { 0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, 0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff }; uint8_t masterKey[Thread::kSizeMasterKey] = { 0 }; EXPECT_EQ(dataset.SetMasterKey(kMasterKey), CHIP_NO_ERROR); EXPECT_EQ(dataset.GetMasterKey(masterKey), CHIP_NO_ERROR); EXPECT_EQ(memcmp(masterKey, kMasterKey, sizeof(kMasterKey)), 0); dataset.UnsetMasterKey(); EXPECT_EQ(dataset.GetMasterKey(masterKey), CHIP_ERROR_TLV_TAG_NOT_FOUND); EXPECT_EQ(dataset.SetMasterKey(masterKey), CHIP_NO_ERROR); } TEST_F(TestThreadOperationalDataset, TestMeshLocalPrefix) { static constexpr uint8_t kMeshLocalPrefix[] = { 0xfd, 0xde, 0xad, 0x00, 0xbe, 0xef, 0x00, 0x00 }; uint8_t meshLocalPrefix[Thread::kSizeMeshLocalPrefix] = { 0 }; EXPECT_EQ(dataset.SetMeshLocalPrefix(kMeshLocalPrefix), CHIP_NO_ERROR); EXPECT_EQ(dataset.GetMeshLocalPrefix(meshLocalPrefix), CHIP_NO_ERROR); EXPECT_EQ(memcmp(meshLocalPrefix, kMeshLocalPrefix, sizeof(kMeshLocalPrefix)), 0); } TEST_F(TestThreadOperationalDataset, TestNetworkName) { static constexpr char kNetworkName[] = "ThreadNetwork"; char networkName[Thread::kSizeNetworkName + 1] = { 0 }; EXPECT_EQ(dataset.SetNetworkName(kNetworkName), CHIP_NO_ERROR); EXPECT_EQ(dataset.GetNetworkName(networkName), CHIP_NO_ERROR); EXPECT_STREQ(networkName, kNetworkName); EXPECT_EQ(dataset.SetNetworkName("0123456789abcdef"), CHIP_NO_ERROR); EXPECT_EQ(dataset.SetNetworkName("0123456789abcdefg"), CHIP_ERROR_INVALID_STRING_LENGTH); EXPECT_EQ(dataset.SetNetworkName(""), CHIP_ERROR_INVALID_STRING_LENGTH); EXPECT_EQ(dataset.SetNetworkName(nullptr), CHIP_ERROR_INVALID_ARGUMENT); } TEST_F(TestThreadOperationalDataset, TestPanId) { static constexpr uint16_t kPanIdValue = 0x1234; uint16_t panid = 0; EXPECT_EQ(dataset.SetPanId(kPanIdValue), CHIP_NO_ERROR); EXPECT_EQ(dataset.GetPanId(panid), CHIP_NO_ERROR); EXPECT_EQ(panid, kPanIdValue); } TEST_F(TestThreadOperationalDataset, TestPSKc) { static constexpr uint8_t kPSKc[] = { 0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, 0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff }; uint8_t pskc[Thread::kSizePSKc] = { 0 }; EXPECT_EQ(dataset.SetPSKc(kPSKc), CHIP_NO_ERROR); EXPECT_EQ(dataset.GetPSKc(pskc), CHIP_NO_ERROR); EXPECT_FALSE(memcmp(pskc, kPSKc, sizeof(kPSKc))); dataset.UnsetPSKc(); EXPECT_EQ(dataset.GetPSKc(pskc), CHIP_ERROR_TLV_TAG_NOT_FOUND); EXPECT_EQ(dataset.SetPSKc(pskc), CHIP_NO_ERROR); } TEST_F(TestThreadOperationalDataset, TestClear) { { EXPECT_EQ(dataset.SetActiveTimestamp(123), CHIP_NO_ERROR); EXPECT_EQ(dataset.SetChannel(5), CHIP_NO_ERROR); uint8_t extendedPanId[Thread::kSizeExtendedPanId] = { 1, 2, 3, 4, 5, 6, 7, 8 }; EXPECT_EQ(dataset.SetExtendedPanId(extendedPanId), CHIP_NO_ERROR); uint8_t masterKey[Thread::kSizeMasterKey] = { 0 }; EXPECT_EQ(dataset.SetMasterKey(masterKey), CHIP_NO_ERROR); uint8_t meshLocalPrefix[Thread::kSizeMeshLocalPrefix] = { 0 }; EXPECT_EQ(dataset.SetMeshLocalPrefix(meshLocalPrefix), CHIP_NO_ERROR); EXPECT_EQ(dataset.SetNetworkName("w00tw00t"), CHIP_NO_ERROR); EXPECT_EQ(dataset.SetPanId(0x4242), CHIP_NO_ERROR); uint8_t pskc[Thread::kSizePSKc] = { 0 }; EXPECT_EQ(dataset.SetPSKc(pskc), CHIP_NO_ERROR); } dataset.Clear(); { uint64_t activeTimestamp; EXPECT_EQ(dataset.GetActiveTimestamp(activeTimestamp), CHIP_ERROR_TLV_TAG_NOT_FOUND); } { uint16_t channel; EXPECT_EQ(dataset.GetChannel(channel), CHIP_ERROR_TLV_TAG_NOT_FOUND); } { uint8_t extendedPanId[Thread::kSizeExtendedPanId] = { 0 }; EXPECT_EQ(dataset.GetExtendedPanId(extendedPanId), CHIP_ERROR_TLV_TAG_NOT_FOUND); } { uint8_t masterKey[Thread::kSizeMasterKey] = { 0 }; EXPECT_EQ(dataset.GetMasterKey(masterKey), CHIP_ERROR_TLV_TAG_NOT_FOUND); } { uint8_t meshLocalPrefix[Thread::kSizeMeshLocalPrefix] = { 0 }; EXPECT_EQ(dataset.GetMeshLocalPrefix(meshLocalPrefix), CHIP_ERROR_TLV_TAG_NOT_FOUND); } { char networkName[Thread::kSizeNetworkName + 1] = { 0 }; EXPECT_EQ(dataset.GetNetworkName(networkName), CHIP_ERROR_TLV_TAG_NOT_FOUND); } { uint16_t panid; EXPECT_EQ(dataset.GetPanId(panid), CHIP_ERROR_TLV_TAG_NOT_FOUND); } { uint8_t pskc[Thread::kSizePSKc] = { 0 }; EXPECT_EQ(dataset.GetPSKc(pskc), CHIP_ERROR_TLV_TAG_NOT_FOUND); } } TEST_F(TestThreadOperationalDataset, TestExampleDataset) { const uint8_t example[] = { 0x0e, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, // Active Timestamp 1 0x00, 0x03, 0x00, 0x00, 0x0f, // Channel 15 0x35, 0x04, 0x07, 0xff, 0xf8, 0x00, // Channel Mask 0x07fff800 0x02, 0x08, 0x39, 0x75, 0x8e, 0xc8, 0x14, 0x4b, 0x07, 0xfb, // Ext PAN ID 39758ec8144b07fb 0x07, 0x08, 0xfd, 0xf1, 0xf1, 0xad, 0xd0, 0x79, 0x7d, 0xc0, // Mesh Local Prefix fdf1:f1ad:d079:7dc0::/64 0x05, 0x10, 0xf3, 0x66, 0xce, 0xc7, 0xa4, 0x46, 0xba, 0xb9, 0x78, 0xd9, 0x0d, 0x27, 0xab, 0xe3, 0x8f, 0x23, // Network Key 0x03, 0x0f, 0x4f, 0x70, 0x65, 0x6e, 0x54, 0x68, 0x72, 0x65, 0x61, 0x64, 0x2d, 0x35, 0x39, 0x33, 0x38, // "OpenThread-5938" 0x01, 0x02, 0x59, 0x38, // PAN ID : 0x5938 0x04, 0x10, 0x3c, 0xa6, 0x7c, 0x96, 0x9e, 0xfb, 0x0d, 0x0c, 0x74, 0xa4, 0xd8, 0xee, 0x92, 0x3b, 0x57, 0x6c, // PKSc 0x0c, 0x04, 0x02, 0xa0, 0xf7, 0xf8, // Security Policy }; EXPECT_EQ(dataset.Init(ByteSpan(example)), CHIP_NO_ERROR); uint64_t activeTimestamp; EXPECT_EQ(dataset.GetActiveTimestamp(activeTimestamp), CHIP_NO_ERROR); EXPECT_EQ(activeTimestamp, 1u); uint16_t channel; EXPECT_EQ(dataset.GetChannel(channel), CHIP_NO_ERROR); EXPECT_EQ(channel, 15u); uint8_t extPanId[Thread::kSizeExtendedPanId]; EXPECT_EQ(dataset.GetExtendedPanId(extPanId), CHIP_NO_ERROR); const uint8_t expectedExtPanId[] = { 0x39, 0x75, 0x8e, 0xc8, 0x14, 0x4b, 0x07, 0xfb }; EXPECT_TRUE(ByteSpan(extPanId).data_equal(ByteSpan(expectedExtPanId))); uint8_t meshLocalPrefix[Thread::kSizeMeshLocalPrefix]; EXPECT_EQ(dataset.GetMeshLocalPrefix(meshLocalPrefix), CHIP_NO_ERROR); const uint8_t expectedMeshLocalPrefix[] = { 0xfd, 0xf1, 0xf1, 0xad, 0xd0, 0x79, 0x7d, 0xc0 }; EXPECT_TRUE(ByteSpan(meshLocalPrefix).data_equal(ByteSpan(expectedMeshLocalPrefix))); uint8_t masterKey[Thread::kSizeMasterKey]; EXPECT_EQ(dataset.GetMasterKey(masterKey), CHIP_NO_ERROR); const uint8_t expectedMasterKey[] = { 0xf3, 0x66, 0xce, 0xc7, 0xa4, 0x46, 0xba, 0xb9, 0x78, 0xd9, 0x0d, 0x27, 0xab, 0xe3, 0x8f, 0x23 }; EXPECT_TRUE(ByteSpan(masterKey).data_equal(ByteSpan(expectedMasterKey))); char networkName[Thread::kSizeNetworkName + 1]; EXPECT_EQ(dataset.GetNetworkName(networkName), CHIP_NO_ERROR); EXPECT_EQ(strncmp(networkName, "OpenThread-5938", sizeof(networkName)), 0); uint16_t panId; EXPECT_EQ(dataset.GetPanId(panId), CHIP_NO_ERROR); EXPECT_EQ(panId, 0x5938u); uint8_t pksc[Thread::kSizePSKc]; EXPECT_EQ(dataset.GetPSKc(pksc), CHIP_NO_ERROR); const uint8_t expectedPksc[] = { 0x3c, 0xa6, 0x7c, 0x96, 0x9e, 0xfb, 0x0d, 0x0c, 0x74, 0xa4, 0xd8, 0xee, 0x92, 0x3b, 0x57, 0x6c }; EXPECT_TRUE(ByteSpan(pksc).data_equal(ByteSpan(expectedPksc))); ByteSpan channelMask; EXPECT_EQ(dataset.GetChannelMask(channelMask), CHIP_NO_ERROR); const uint8_t expectedChannelMask[] = { 0x07, 0xff, 0xf8, 0x00 }; EXPECT_TRUE(channelMask.data_equal(ByteSpan(expectedChannelMask))); uint32_t securityPolicy; EXPECT_EQ(dataset.GetSecurityPolicy(securityPolicy), CHIP_NO_ERROR); EXPECT_EQ(securityPolicy, 0x02a0f7f8u); } TEST_F(TestThreadOperationalDataset, TestInvalidExampleDataset) { const uint8_t invalid[] = { 0x0e, 0x01, 0x01, // Active Timestamp 0x00, 0x01, 0x0f, // Channel 0x35, 0x00, // Channel Mask 0x02, 0x09, 0x39, 0x75, 0x8e, 0xc8, 0x14, 0x4b, 0x07, 0xfb, 0xff, // Ext PAN ID 0x07, 0x01, 0x01, // Mesh Local Prefix 0x05, 0x0f, 0xf3, 0x66, 0xce, 0xc7, 0xa4, 0x46, 0xba, 0xb9, 0x78, 0xd9, 0x0d, 0x27, 0xab, 0xe3, 0x8f, // Network Key 0x03, 0x11, 0x4f, 0x70, 0x65, 0x6e, 0x54, 0x68, 0x72, 0x65, 0x61, 0x64, 0x2d, 0x35, 0x39, 0x33, 0x38, 0xff, 0xff, // Name 0x01, 0x01, 0xff, // PAN ID 0x04, 0x01, 0x3c, // PKSc 0x0c, 0x01, 0x01, // Security Policy }; // The overall TLV structure is valid, but all TLVs have invalid sizes EXPECT_EQ(dataset.Init(ByteSpan(invalid)), CHIP_NO_ERROR); uint64_t activeTimestamp; EXPECT_EQ(dataset.GetActiveTimestamp(activeTimestamp), CHIP_ERROR_INVALID_TLV_ELEMENT); uint16_t channel; EXPECT_EQ(dataset.GetChannel(channel), CHIP_ERROR_INVALID_TLV_ELEMENT); ByteSpan extPanId; EXPECT_EQ(dataset.GetExtendedPanIdAsByteSpan(extPanId), CHIP_ERROR_INVALID_TLV_ELEMENT); uint8_t meshLocalPrefix[Thread::kSizeMeshLocalPrefix]; EXPECT_EQ(dataset.GetMeshLocalPrefix(meshLocalPrefix), CHIP_ERROR_INVALID_TLV_ELEMENT); uint8_t masterKey[Thread::kSizeMasterKey]; EXPECT_EQ(dataset.GetMasterKey(masterKey), CHIP_ERROR_INVALID_TLV_ELEMENT); char networkName[Thread::kSizeNetworkName + 1]; EXPECT_EQ(dataset.GetNetworkName(networkName), CHIP_ERROR_INVALID_TLV_ELEMENT); uint16_t panId; EXPECT_EQ(dataset.GetPanId(panId), CHIP_ERROR_INVALID_TLV_ELEMENT); uint8_t pksc[Thread::kSizePSKc]; EXPECT_EQ(dataset.GetPSKc(pksc), CHIP_ERROR_INVALID_TLV_ELEMENT); ByteSpan channelMask; EXPECT_EQ(dataset.GetChannelMask(channelMask), CHIP_ERROR_INVALID_TLV_ELEMENT); uint32_t securityPolicy; EXPECT_EQ(dataset.GetSecurityPolicy(securityPolicy), CHIP_ERROR_INVALID_TLV_ELEMENT); } } // namespace