blob: c87584e790b5275893a2999448ce5bda4e124274 [file] [log] [blame]
/*
* Copyright (c) 2020 Project CHIP Authors
* All rights reserved.
*
* 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 "Command.h"
#include <netdb.h>
#include <sstream>
#include <sys/socket.h>
#include <sys/types.h>
#include <support/ErrorMacros.h>
#include <support/SafeInt.h>
#include <support/logging/CHIPLogging.h>
bool Command::InitArguments(int argc, char ** argv)
{
bool isValidCommand = false;
size_t argsCount = mArgs.size();
VerifyOrExit(argsCount == (size_t)(argc),
ChipLogError(chipTool, "InitArgs: Wrong arguments number: %zu instead of %zu", argc, argsCount));
for (size_t i = 0; i < argsCount; i++)
{
if (!InitArgument(i, argv[i]))
{
ExitNow();
}
}
isValidCommand = true;
exit:
return isValidCommand;
}
static bool ParseAddressWithInterface(const char * addressString, Command::AddressWithInterface * address)
{
struct addrinfo hints;
struct addrinfo * result;
int ret;
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_DGRAM;
ret = getaddrinfo(addressString, nullptr, &hints, &result);
if (ret < 0)
{
ChipLogError(chipTool, "Invalid address: %s", addressString);
return false;
}
address->address = ::chip::Inet::IPAddress::FromSockAddr(*result->ai_addr);
if (result->ai_family == AF_INET6)
{
struct sockaddr_in6 * addr = reinterpret_cast<struct sockaddr_in6 *>(result->ai_addr);
address->interfaceId = addr->sin6_scope_id;
}
else
{
address->interfaceId = INET_NULL_INTERFACEID;
}
return true;
}
bool Command::InitArgument(size_t argIndex, const char * argValue)
{
bool isValidArgument = false;
Argument arg = mArgs.at(argIndex);
switch (arg.type)
{
case ArgumentType::Attribute: {
char * value = reinterpret_cast<char *>(arg.value);
isValidArgument = (strcmp(argValue, value) == 0);
break;
}
case ArgumentType::String: {
const char ** value = reinterpret_cast<const char **>(arg.value);
*value = const_cast<char *>(argValue);
isValidArgument = (strcmp(argValue, *value) == 0);
break;
}
case ArgumentType::Number_uint8: {
uint8_t * value = reinterpret_cast<uint8_t *>(arg.value);
// stringstream treats uint8_t as char, which is not what we want here.
uint16_t tmpValue;
std::stringstream ss(argValue);
ss >> tmpValue;
if (chip::CanCastTo<uint8_t>(tmpValue))
{
*value = static_cast<uint8_t>(tmpValue);
uint64_t min = chip::CanCastTo<uint64_t>(arg.min) ? static_cast<uint64_t>(arg.min) : 0;
uint64_t max = arg.max;
isValidArgument = (!ss.fail() && ss.eof() && *value >= min && *value <= max);
}
else
{
isValidArgument = false;
}
break;
}
case ArgumentType::Number_uint16: {
uint16_t * value = reinterpret_cast<uint16_t *>(arg.value);
std::stringstream ss(argValue);
ss >> *value;
uint64_t min = chip::CanCastTo<uint64_t>(arg.min) ? static_cast<uint64_t>(arg.min) : 0;
uint64_t max = arg.max;
isValidArgument = (!ss.fail() && ss.eof() && *value >= min && *value <= max);
break;
}
case ArgumentType::Number_uint32: {
uint32_t * value = reinterpret_cast<uint32_t *>(arg.value);
std::stringstream ss(argValue);
ss >> *value;
uint64_t min = chip::CanCastTo<uint64_t>(arg.min) ? static_cast<uint64_t>(arg.min) : 0;
uint64_t max = arg.max;
isValidArgument = (!ss.fail() && ss.eof() && *value >= min && *value <= max);
break;
}
case ArgumentType::Number_uint64: {
uint64_t * value = reinterpret_cast<uint64_t *>(arg.value);
std::stringstream ss(argValue);
ss >> *value;
uint64_t min = chip::CanCastTo<uint64_t>(arg.min) ? static_cast<uint64_t>(arg.min) : 0;
uint64_t max = arg.max;
isValidArgument = (!ss.fail() && ss.eof() && *value >= min && *value <= max);
break;
}
case ArgumentType::Number_int8: {
int8_t * value = reinterpret_cast<int8_t *>(arg.value);
// stringstream treats int8_t as char, which is not what we want here.
int16_t tmpValue;
std::stringstream ss(argValue);
ss >> tmpValue;
if (chip::CanCastTo<int8_t>(tmpValue))
{
*value = static_cast<int8_t>(tmpValue);
int64_t min = arg.min;
int64_t max = chip::CanCastTo<int64_t>(arg.max) ? static_cast<int64_t>(arg.max) : INT64_MAX;
isValidArgument = (!ss.fail() && ss.eof() && *value >= min && *value <= max);
}
else
{
isValidArgument = false;
}
break;
}
case ArgumentType::Number_int16: {
int16_t * value = reinterpret_cast<int16_t *>(arg.value);
std::stringstream ss(argValue);
ss >> *value;
int64_t min = arg.min;
int64_t max = chip::CanCastTo<int64_t>(arg.max) ? static_cast<int64_t>(arg.max) : INT64_MAX;
isValidArgument = (!ss.fail() && ss.eof() && *value >= min && *value <= max);
break;
}
case ArgumentType::Number_int32: {
int32_t * value = reinterpret_cast<int32_t *>(arg.value);
std::stringstream ss(argValue);
ss >> *value;
int64_t min = arg.min;
int64_t max = chip::CanCastTo<int64_t>(arg.max) ? static_cast<int64_t>(arg.max) : INT64_MAX;
isValidArgument = (!ss.fail() && ss.eof() && *value >= min && *value <= max);
break;
}
case ArgumentType::Number_int64: {
int64_t * value = reinterpret_cast<int64_t *>(arg.value);
std::stringstream ss(argValue);
ss >> *value;
int64_t min = arg.min;
int64_t max = chip::CanCastTo<int64_t>(arg.max) ? static_cast<int64_t>(arg.max) : INT64_MAX;
isValidArgument = (!ss.fail() && ss.eof() && *value >= min && *value <= max);
break;
}
case ArgumentType::Address: {
AddressWithInterface * value = reinterpret_cast<AddressWithInterface *>(arg.value);
isValidArgument = ParseAddressWithInterface(argValue, value);
break;
}
}
if (!isValidArgument)
{
ChipLogError(chipTool, "InitArgs: Invalid argument %s: %s", arg.name, argValue);
}
return isValidArgument;
}
size_t Command::AddArgument(const char * name, const char * value)
{
Argument arg;
arg.type = ArgumentType::Attribute;
arg.name = name;
arg.value = const_cast<void *>(reinterpret_cast<const void *>(value));
mArgs.emplace_back(arg);
return mArgs.size();
}
size_t Command::AddArgument(const char * name, char ** value)
{
Argument arg;
arg.type = ArgumentType::String;
arg.name = name;
arg.value = reinterpret_cast<void *>(value);
mArgs.emplace_back(arg);
return mArgs.size();
}
size_t Command::AddArgument(const char * name, AddressWithInterface * out)
{
Argument arg;
arg.type = ArgumentType::Address;
arg.name = name;
arg.value = reinterpret_cast<void *>(out);
mArgs.emplace_back(arg);
return mArgs.size();
}
size_t Command::AddArgument(const char * name, int64_t min, uint64_t max, void * out, ArgumentType type)
{
Argument arg;
arg.type = type;
arg.name = name;
arg.value = out;
arg.min = min;
arg.max = max;
mArgs.emplace_back(arg);
return mArgs.size();
}
size_t Command::AddArgument(const char * name, int64_t min, uint64_t max, void * out)
{
Argument arg;
arg.type = ArgumentType::Number_uint8;
arg.name = name;
arg.value = out;
arg.min = min;
arg.max = max;
mArgs.emplace_back(arg);
return mArgs.size();
}
const char * Command::GetArgumentName(size_t index) const
{
if (index < mArgs.size())
{
return mArgs.at(index).name;
}
return nullptr;
}
const char * Command::GetAttribute(void) const
{
size_t argsCount = mArgs.size();
for (size_t i = 0; i < argsCount; i++)
{
Argument arg = mArgs.at(i);
if (arg.type == ArgumentType::Attribute)
{
return reinterpret_cast<const char *>(arg.value);
}
}
return nullptr;
}
void Command::UpdateWaitForResponse(bool value)
{
{
std::lock_guard<std::mutex> lk(cvWaitingForResponseMutex);
mWaitingForResponse = value;
}
cvWaitingForResponse.notify_all();
}
void Command::WaitForResponse(uint16_t duration)
{
std::chrono::seconds waitingForResponseTimeout(duration);
std::unique_lock<std::mutex> lk(cvWaitingForResponseMutex);
auto waitingUntil = std::chrono::system_clock::now() + waitingForResponseTimeout;
if (!cvWaitingForResponse.wait_until(lk, waitingUntil, [this]() { return !this->mWaitingForResponse; }))
{
ChipLogError(chipTool, "No response from device");
}
}