699 lines
21 KiB
C++
699 lines
21 KiB
C++
/*
|
|
* Copyright (c) 2016, The OpenThread Authors.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. Neither the name of the copyright holder nor the
|
|
* names of its contributors may be used to endorse or promote products
|
|
* derived from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
/**
|
|
* @file
|
|
* This file implements the CLI interpreter.
|
|
*/
|
|
|
|
#ifdef OPENTHREAD_CONFIG_FILE
|
|
#include OPENTHREAD_CONFIG_FILE
|
|
#else
|
|
#include <openthread-config.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <openthread.h>
|
|
|
|
#include "cli.hpp"
|
|
#include "cli_dataset.hpp"
|
|
|
|
namespace Thread {
|
|
namespace Cli {
|
|
|
|
const DatasetCommand Dataset::sCommands[] =
|
|
{
|
|
{ "help", &ProcessHelp },
|
|
{ "active", &ProcessActive },
|
|
{ "activetimestamp", &ProcessActiveTimestamp },
|
|
{ "channel", &ProcessChannel },
|
|
{ "clear", &ProcessClear },
|
|
{ "commit", &ProcessCommit },
|
|
{ "delay", &ProcessDelay },
|
|
{ "extpanid", &ProcessExtPanId },
|
|
{ "masterkey", &ProcessMasterKey },
|
|
{ "meshlocalprefix", &ProcessMeshLocalPrefix },
|
|
{ "mgmtgetcommand", &ProcessMgmtGetCommand },
|
|
{ "mgmtsetcommand", &ProcessMgmtSetCommand },
|
|
{ "networkname", &ProcessNetworkName },
|
|
{ "panid", &ProcessPanId },
|
|
{ "pending", &ProcessPending },
|
|
{ "pendingtimestamp", &ProcessPendingTimestamp },
|
|
{ "pskc", &ProcessPSKc },
|
|
{ "securitypolicy", &ProcessSecurityPolicy },
|
|
};
|
|
|
|
Server *Dataset::sServer;
|
|
otOperationalDataset Dataset::sDataset;
|
|
|
|
void Dataset::OutputBytes(const uint8_t *aBytes, uint8_t aLength)
|
|
{
|
|
for (int i = 0; i < aLength; i++)
|
|
{
|
|
sServer->OutputFormat("%02x", aBytes[i]);
|
|
}
|
|
}
|
|
|
|
ThreadError Dataset::Print(otOperationalDataset &aDataset)
|
|
{
|
|
if (aDataset.mIsPendingTimestampSet)
|
|
{
|
|
sServer->OutputFormat("Pending Timestamp: %d\r\n", aDataset.mPendingTimestamp);
|
|
}
|
|
|
|
if (aDataset.mIsActiveTimestampSet)
|
|
{
|
|
sServer->OutputFormat("Active Timestamp: %d\r\n", aDataset.mActiveTimestamp);
|
|
}
|
|
|
|
if (aDataset.mIsChannelSet)
|
|
{
|
|
sServer->OutputFormat("Channel: %d\r\n", aDataset.mChannel);
|
|
}
|
|
|
|
if (aDataset.mIsDelaySet)
|
|
{
|
|
sServer->OutputFormat("Delay: %d\r\n", aDataset.mDelay);
|
|
}
|
|
|
|
if (aDataset.mIsExtendedPanIdSet)
|
|
{
|
|
sServer->OutputFormat("Ext PAN ID: ");
|
|
OutputBytes(aDataset.mExtendedPanId.m8, sizeof(aDataset.mExtendedPanId));
|
|
sServer->OutputFormat("\r\n");
|
|
}
|
|
|
|
if (aDataset.mIsMeshLocalPrefixSet)
|
|
{
|
|
const uint8_t *prefix = aDataset.mMeshLocalPrefix.m8;
|
|
sServer->OutputFormat("Mesh Local Prefix: %x:%x:%x:%x/64\r\n",
|
|
(static_cast<uint16_t>(prefix[0]) << 8) | prefix[1],
|
|
(static_cast<uint16_t>(prefix[2]) << 8) | prefix[3],
|
|
(static_cast<uint16_t>(prefix[4]) << 8) | prefix[5],
|
|
(static_cast<uint16_t>(prefix[6]) << 8) | prefix[7]);
|
|
}
|
|
|
|
if (aDataset.mIsMasterKeySet)
|
|
{
|
|
sServer->OutputFormat("Master Key: ");
|
|
OutputBytes(aDataset.mMasterKey.m8, sizeof(aDataset.mMasterKey));
|
|
sServer->OutputFormat("\r\n");
|
|
}
|
|
|
|
if (aDataset.mIsNetworkNameSet)
|
|
{
|
|
sServer->OutputFormat("Network Name: ");
|
|
sServer->OutputFormat("%.*s\r\n", sizeof(aDataset.mNetworkName), aDataset.mNetworkName.m8);
|
|
}
|
|
|
|
if (aDataset.mIsPanIdSet)
|
|
{
|
|
sServer->OutputFormat("PAN ID: 0x%04x\r\n", aDataset.mPanId);
|
|
}
|
|
|
|
if (aDataset.mIsPSKcSet)
|
|
{
|
|
sServer->OutputFormat("PSKc: ");
|
|
OutputBytes(aDataset.mPSKc.m8, sizeof(aDataset.mPSKc.m8));
|
|
sServer->OutputFormat("\r\n");
|
|
}
|
|
|
|
if (aDataset.mIsSecurityPolicySet)
|
|
{
|
|
sServer->OutputFormat("Security Policy: %d, ", aDataset.mSecurityPolicy.mRotationTime);
|
|
|
|
if (aDataset.mSecurityPolicy.mFlags & OT_SECURITY_POLICY_OBTAIN_MASTER_KEY)
|
|
{
|
|
sServer->OutputFormat("o");
|
|
}
|
|
|
|
if (aDataset.mSecurityPolicy.mFlags & OT_SECURITY_POLICY_NATIVE_COMMISSIONING)
|
|
{
|
|
sServer->OutputFormat("n");
|
|
}
|
|
|
|
if (aDataset.mSecurityPolicy.mFlags & OT_SECURITY_POLICY_ROUTERS)
|
|
{
|
|
sServer->OutputFormat("r");
|
|
}
|
|
|
|
if (aDataset.mSecurityPolicy.mFlags & OT_SECURITY_POLICY_EXTERNAL_COMMISSIONER)
|
|
{
|
|
sServer->OutputFormat("c");
|
|
}
|
|
|
|
if (aDataset.mSecurityPolicy.mFlags & OT_SECURITY_POLICY_BEACONS)
|
|
{
|
|
sServer->OutputFormat("b");
|
|
}
|
|
|
|
sServer->OutputFormat("\r\n");
|
|
}
|
|
|
|
return kThreadError_None;
|
|
}
|
|
|
|
ThreadError Dataset::Process(otInstance *aInstance, int argc, char *argv[], Server &aServer)
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
|
|
sServer = &aServer;
|
|
|
|
if (argc == 0)
|
|
{
|
|
ExitNow(error = Print(sDataset));
|
|
}
|
|
|
|
for (unsigned int i = 0; i < sizeof(sCommands) / sizeof(sCommands[0]); i++)
|
|
{
|
|
if (strcmp(argv[0], sCommands[i].mName) == 0)
|
|
{
|
|
error = sCommands[i].mCommand(aInstance, argc - 1, argv + 1);
|
|
break;
|
|
}
|
|
}
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessHelp(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
for (unsigned int i = 0; i < sizeof(sCommands) / sizeof(sCommands[0]); i++)
|
|
{
|
|
sServer->OutputFormat("%s\r\n", sCommands[i].mName);
|
|
}
|
|
|
|
(void)aInstance;
|
|
(void)argc;
|
|
(void)argv;
|
|
return kThreadError_None;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessActive(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
otOperationalDataset dataset;
|
|
otGetActiveDataset(aInstance, &dataset);
|
|
|
|
(void)argc;
|
|
(void)argv;
|
|
return Print(dataset);
|
|
}
|
|
|
|
ThreadError Dataset::ProcessActiveTimestamp(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
long value;
|
|
|
|
VerifyOrExit(argc > 0, ;);
|
|
|
|
SuccessOrExit(error = Interpreter::ParseLong(argv[0], value));
|
|
sDataset.mActiveTimestamp = static_cast<uint64_t>(value);
|
|
sDataset.mIsActiveTimestampSet = true;
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessChannel(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
long value;
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
SuccessOrExit(error = Interpreter::ParseLong(argv[0], value));
|
|
sDataset.mChannel = static_cast<uint16_t>(value);
|
|
sDataset.mIsChannelSet = true;
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessClear(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
memset(&sDataset, 0, sizeof(sDataset));
|
|
(void)aInstance;
|
|
(void)argc;
|
|
(void)argv;
|
|
return kThreadError_None;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessCommit(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
|
|
VerifyOrExit(argc > 0, ;);
|
|
|
|
if (strcmp(argv[0], "active") == 0)
|
|
{
|
|
SuccessOrExit(error = otSetActiveDataset(aInstance, &sDataset));
|
|
}
|
|
else if (strcmp(argv[0], "pending") == 0)
|
|
{
|
|
SuccessOrExit(error = otSetPendingDataset(aInstance, &sDataset));
|
|
}
|
|
else
|
|
{
|
|
ExitNow(error = kThreadError_Parse);
|
|
}
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessDelay(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
long value;
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
SuccessOrExit(error = Interpreter::ParseLong(argv[0], value));
|
|
sDataset.mDelay = static_cast<uint32_t>(value);
|
|
sDataset.mIsDelaySet = true;
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessExtPanId(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
uint8_t extPanId[OT_EXT_PAN_ID_SIZE];
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
VerifyOrExit(Interpreter::Hex2Bin(argv[0], extPanId, sizeof(extPanId)) >= 0, error = kThreadError_Parse);
|
|
|
|
memcpy(sDataset.mExtendedPanId.m8, extPanId, sizeof(sDataset.mExtendedPanId));
|
|
sDataset.mIsExtendedPanIdSet = true;
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessMasterKey(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
int keyLength;
|
|
uint8_t key[OT_MASTER_KEY_SIZE];
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
VerifyOrExit((keyLength = Interpreter::Hex2Bin(argv[0], key, sizeof(key))) == OT_MASTER_KEY_SIZE,
|
|
error = kThreadError_Parse);
|
|
|
|
memcpy(sDataset.mMasterKey.m8, key, sizeof(sDataset.mMasterKey));
|
|
sDataset.mIsMasterKeySet = true;
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessMeshLocalPrefix(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
otIp6Address prefix;
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
SuccessOrExit(error = otIp6AddressFromString(argv[0], &prefix));
|
|
|
|
memcpy(sDataset.mMeshLocalPrefix.m8, prefix.mFields.m8, sizeof(sDataset.mMeshLocalPrefix.m8));
|
|
sDataset.mIsMeshLocalPrefixSet = true;
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessNetworkName(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
size_t length;
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
VerifyOrExit((length = strlen(argv[0])) <= OT_NETWORK_NAME_MAX_SIZE, error = kThreadError_Parse);
|
|
|
|
memset(&sDataset.mNetworkName, 0, sizeof(sDataset.mNetworkName));
|
|
memcpy(sDataset.mNetworkName.m8, argv[0], length);
|
|
sDataset.mIsNetworkNameSet = true;
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessPanId(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
long value;
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
SuccessOrExit(error = Interpreter::ParseLong(argv[0], value));
|
|
sDataset.mPanId = static_cast<otPanId>(value);
|
|
sDataset.mIsPanIdSet = true;
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessPending(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
otOperationalDataset dataset;
|
|
otGetPendingDataset(aInstance, &dataset);
|
|
|
|
(void)argc;
|
|
(void)argv;
|
|
return Print(dataset);
|
|
}
|
|
|
|
ThreadError Dataset::ProcessPendingTimestamp(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
long value;
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
SuccessOrExit(error = Interpreter::ParseLong(argv[0], value));
|
|
sDataset.mPendingTimestamp = static_cast<uint64_t>(value);
|
|
sDataset.mIsPendingTimestampSet = true;
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessMgmtSetCommand(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
otOperationalDataset dataset;
|
|
uint8_t tlvs[32];
|
|
long value;
|
|
int length = 0;
|
|
otIp6Address prefix;
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
|
|
memset(&dataset, 0, sizeof(dataset));
|
|
|
|
for (uint8_t index = 1; index < argc; index++)
|
|
{
|
|
if (strcmp(argv[index], "activetimestamp") == 0)
|
|
{
|
|
VerifyOrExit(index < argc, error = kThreadError_Parse);
|
|
dataset.mIsActiveTimestampSet = true;
|
|
SuccessOrExit(error = Interpreter::ParseLong(argv[++index], value));
|
|
dataset.mActiveTimestamp = static_cast<uint64_t>(value);
|
|
}
|
|
else if (strcmp(argv[index], "pendingtimestamp") == 0)
|
|
{
|
|
VerifyOrExit(index < argc, error = kThreadError_Parse);
|
|
dataset.mIsPendingTimestampSet = true;
|
|
SuccessOrExit(error = Interpreter::ParseLong(argv[++index], value));
|
|
dataset.mPendingTimestamp = static_cast<uint64_t>(value);
|
|
}
|
|
else if (strcmp(argv[index], "masterkey") == 0)
|
|
{
|
|
VerifyOrExit(index < argc, error = kThreadError_Parse);
|
|
dataset.mIsMasterKeySet = true;
|
|
VerifyOrExit((length = Interpreter::Hex2Bin(argv[++index], dataset.mMasterKey.m8,
|
|
sizeof(dataset.mMasterKey.m8))) == OT_MASTER_KEY_SIZE, error = kThreadError_Parse);
|
|
length = 0;
|
|
}
|
|
else if (strcmp(argv[index], "networkname") == 0)
|
|
{
|
|
VerifyOrExit(index < argc, error = kThreadError_Parse);
|
|
dataset.mIsNetworkNameSet = true;
|
|
VerifyOrExit((length = static_cast<int>(strlen(argv[++index]))) <= OT_NETWORK_NAME_MAX_SIZE,
|
|
error = kThreadError_Parse);
|
|
memset(&dataset.mNetworkName, 0, sizeof(sDataset.mNetworkName));
|
|
memcpy(dataset.mNetworkName.m8, argv[0], static_cast<size_t>(length));
|
|
length = 0;
|
|
}
|
|
else if (strcmp(argv[index], "extpanid") == 0)
|
|
{
|
|
VerifyOrExit(index < argc, error = kThreadError_Parse);
|
|
dataset.mIsExtendedPanIdSet = true;
|
|
VerifyOrExit(Interpreter::Hex2Bin(argv[++index], dataset.mExtendedPanId.m8,
|
|
sizeof(dataset.mExtendedPanId.m8)) >= 0, error = kThreadError_Parse);
|
|
}
|
|
else if (strcmp(argv[index], "localprefix") == 0)
|
|
{
|
|
VerifyOrExit(index < argc, error = kThreadError_Parse);
|
|
dataset.mIsMeshLocalPrefixSet = true;
|
|
SuccessOrExit(error = otIp6AddressFromString(argv[++index], &prefix));
|
|
memcpy(dataset.mMeshLocalPrefix.m8, prefix.mFields.m8, sizeof(dataset.mMeshLocalPrefix.m8));
|
|
}
|
|
else if (strcmp(argv[index], "delaytimer") == 0)
|
|
{
|
|
VerifyOrExit(index < argc, error = kThreadError_Parse);
|
|
dataset.mIsDelaySet = true;
|
|
SuccessOrExit(error = Interpreter::ParseLong(argv[++index], value));
|
|
dataset.mDelay = static_cast<uint32_t>(value);
|
|
}
|
|
else if (strcmp(argv[index], "panid") == 0)
|
|
{
|
|
VerifyOrExit(index < argc, error = kThreadError_Parse);
|
|
dataset.mIsPanIdSet = true;
|
|
SuccessOrExit(error = Interpreter::ParseLong(argv[++index], value));
|
|
dataset.mPanId = static_cast<otPanId>(value);
|
|
}
|
|
else if (strcmp(argv[index], "channel") == 0)
|
|
{
|
|
VerifyOrExit(index < argc, error = kThreadError_Parse);
|
|
dataset.mIsChannelSet = true;
|
|
SuccessOrExit(error = Interpreter::ParseLong(argv[++index], value));
|
|
dataset.mChannel = static_cast<uint16_t>(value);
|
|
}
|
|
else if (strcmp(argv[index], "binary") == 0)
|
|
{
|
|
VerifyOrExit((index + 1) < argc, error = kThreadError_Parse);
|
|
length = static_cast<int>((strlen(argv[++index]) + 1) / 2);
|
|
VerifyOrExit(static_cast<size_t>(length) <= sizeof(tlvs), error = kThreadError_NoBufs);
|
|
VerifyOrExit(Interpreter::Hex2Bin(argv[index], tlvs, static_cast<uint16_t>(length)) >= 0,
|
|
error = kThreadError_Parse);
|
|
}
|
|
else
|
|
{
|
|
ExitNow(error = kThreadError_Parse);
|
|
}
|
|
}
|
|
|
|
if (strcmp(argv[0], "active") == 0)
|
|
{
|
|
SuccessOrExit(error = otSendActiveSet(aInstance, &dataset, tlvs, static_cast<uint8_t>(length)));
|
|
}
|
|
else if (strcmp(argv[0], "pending") == 0)
|
|
{
|
|
SuccessOrExit(error = otSendPendingSet(aInstance, &dataset, tlvs, static_cast<uint8_t>(length)));
|
|
}
|
|
else
|
|
{
|
|
ExitNow(error = kThreadError_Parse);
|
|
}
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessMgmtGetCommand(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
otOperationalDataset dataset;
|
|
uint8_t tlvs[32];
|
|
long value;
|
|
int length = 0;
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
|
|
memset(&dataset, 0, sizeof(dataset));
|
|
|
|
for (uint8_t index = 1; index < argc; index++)
|
|
{
|
|
VerifyOrExit(static_cast<size_t>(length) < sizeof(tlvs), error = kThreadError_NoBufs);
|
|
|
|
if (strcmp(argv[index], "activetimestamp") == 0)
|
|
{
|
|
tlvs[length++] = OT_MESHCOP_TLV_ACTIVETIMESTAMP;
|
|
}
|
|
else if (strcmp(argv[index], "pendingtimestamp") == 0)
|
|
{
|
|
tlvs[length++] = OT_MESHCOP_TLV_PENDINGTIMESTAMP;
|
|
}
|
|
else if (strcmp(argv[index], "masterkey") == 0)
|
|
{
|
|
tlvs[length++] = OT_MESHCOP_TLV_MASTERKEY;
|
|
}
|
|
else if (strcmp(argv[index], "networkname") == 0)
|
|
{
|
|
tlvs[length++] = OT_MESHCOP_TLV_NETWORKNAME;
|
|
}
|
|
else if (strcmp(argv[index], "extpanid") == 0)
|
|
{
|
|
tlvs[length++] = OT_MESHCOP_TLV_EXTPANID;
|
|
}
|
|
else if (strcmp(argv[index], "localprefix") == 0)
|
|
{
|
|
tlvs[length++] = OT_MESHCOP_TLV_MESHLOCALPREFIX;
|
|
}
|
|
else if (strcmp(argv[index], "delaytimer") == 0)
|
|
{
|
|
tlvs[length++] = OT_MESHCOP_TLV_DELAYTIMER;
|
|
}
|
|
else if (strcmp(argv[index], "panid") == 0)
|
|
{
|
|
tlvs[length++] = OT_MESHCOP_TLV_PANID;
|
|
}
|
|
else if (strcmp(argv[index], "channel") == 0)
|
|
{
|
|
tlvs[length++] = OT_MESHCOP_TLV_CHANNEL;
|
|
}
|
|
else if (strcmp(argv[index], "binary") == 0)
|
|
{
|
|
VerifyOrExit((index + 1) < argc, error = kThreadError_Parse);
|
|
value = static_cast<long>(strlen(argv[++index]) + 1) / 2;
|
|
VerifyOrExit(static_cast<size_t>(value) <= (sizeof(tlvs) - static_cast<size_t>(length)), error = kThreadError_NoBufs);
|
|
VerifyOrExit(Interpreter::Hex2Bin(argv[index], tlvs + length, static_cast<uint16_t>(value)) >= 0,
|
|
error = kThreadError_Parse);
|
|
length += value;
|
|
}
|
|
else
|
|
{
|
|
ExitNow(error = kThreadError_Parse);
|
|
}
|
|
}
|
|
|
|
if (strcmp(argv[0], "active") == 0)
|
|
{
|
|
SuccessOrExit(error = otSendActiveGet(aInstance, tlvs, static_cast<uint8_t>(length)));
|
|
}
|
|
else if (strcmp(argv[0], "pending") == 0)
|
|
{
|
|
SuccessOrExit(error = otSendPendingGet(aInstance, tlvs, static_cast<uint8_t>(length)));
|
|
}
|
|
else
|
|
{
|
|
ExitNow(error = kThreadError_Parse);
|
|
}
|
|
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessPSKc(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
uint16_t length;
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
|
|
length = static_cast<uint16_t>((strlen(argv[0]) + 1) / 2);
|
|
VerifyOrExit(length <= OT_PSKC_MAX_SIZE, error = kThreadError_NoBufs);
|
|
VerifyOrExit(Interpreter::Hex2Bin(argv[0], sDataset.mPSKc.m8 + OT_PSKC_MAX_SIZE - length, length)
|
|
== length, error = kThreadError_Parse);
|
|
|
|
sDataset.mIsPSKcSet = true;
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
ThreadError Dataset::ProcessSecurityPolicy(otInstance *aInstance, int argc, char *argv[])
|
|
{
|
|
ThreadError error = kThreadError_None;
|
|
long value;
|
|
|
|
VerifyOrExit(argc > 0, error = kThreadError_Parse);
|
|
|
|
SuccessOrExit(error = Interpreter::ParseLong(argv[0], value));
|
|
sDataset.mSecurityPolicy.mRotationTime = static_cast<uint16_t>(value);
|
|
sDataset.mSecurityPolicy.mFlags = 0;
|
|
|
|
if (argc > 1)
|
|
{
|
|
for (char *arg = argv[1]; *arg != '\0'; arg++)
|
|
{
|
|
switch (*arg)
|
|
{
|
|
case 'o':
|
|
sDataset.mSecurityPolicy.mFlags |= OT_SECURITY_POLICY_OBTAIN_MASTER_KEY;
|
|
break;
|
|
|
|
case 'n':
|
|
sDataset.mSecurityPolicy.mFlags |= OT_SECURITY_POLICY_NATIVE_COMMISSIONING;
|
|
break;
|
|
|
|
case 'r':
|
|
sDataset.mSecurityPolicy.mFlags |= OT_SECURITY_POLICY_ROUTERS;
|
|
break;
|
|
|
|
case 'c':
|
|
sDataset.mSecurityPolicy.mFlags |= OT_SECURITY_POLICY_EXTERNAL_COMMISSIONER;
|
|
break;
|
|
|
|
case 'b':
|
|
sDataset.mSecurityPolicy.mFlags |= OT_SECURITY_POLICY_BEACONS;
|
|
break;
|
|
|
|
default:
|
|
ExitNow(error = kThreadError_Parse);
|
|
}
|
|
}
|
|
}
|
|
|
|
sDataset.mIsSecurityPolicySet = true;
|
|
(void)aInstance;
|
|
|
|
exit:
|
|
return error;
|
|
}
|
|
|
|
} // namespace Cli
|
|
} // namespace Thread
|