blob: 2ea4d3c99eba305e949487276aed3b2205731803 [file] [log] [blame]
/*
*
* 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 <pw_unit_test/framework.h>
#include <lib/core/StringBuilderAdapters.h>
#include <lib/support/CHIPMem.h>
#include <lib/support/ThreadOperationalDataset.h>
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