blob: 9a9ba96598c9729c3887f23b26086cab6ea1bc6d [file] [log] [blame]
/**
*
* Copyright (c) 2020 Project CHIP Authors
* Copyright (c) 2018 Google LLC.
* Copyright (c) 2016-2017 Nest Labs, Inc.
* 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.
*/
/**
* @file
* This file defines Status Information Block in Interaction Model
*
*/
#include "StatusIB.h"
#include "MessageDefHelper.h"
#include <inttypes.h>
#include <stdarg.h>
#include <stdio.h>
#include <app/AppBuildConfig.h>
#include <lib/core/CHIPCore.h>
#include <lib/support/ErrorStr.h>
using namespace chip;
using namespace chip::TLV;
using namespace chip::Protocols::InteractionModel;
namespace chip {
namespace app {
CHIP_ERROR StatusIB::Parser::DecodeStatusIB(StatusIB & aStatusIB) const
{
TLV::TLVReader reader;
reader.Init(mReader);
while (CHIP_NO_ERROR == reader.Next())
{
if (!TLV::IsContextTag(reader.GetTag()))
{
continue;
}
switch (TLV::TagNumFromTag(reader.GetTag()))
{
case to_underlying(Tag::kStatus):
ReturnErrorOnFailure(reader.Get(aStatusIB.mStatus));
break;
case to_underlying(Tag::kClusterStatus):
ClusterStatus clusterStatus;
ReturnErrorOnFailure(reader.Get(clusterStatus));
aStatusIB.mClusterStatus.SetValue(clusterStatus);
break;
}
}
return CHIP_NO_ERROR;
}
#if CHIP_CONFIG_IM_ENABLE_SCHEMA_CHECK
CHIP_ERROR StatusIB::Parser::CheckSchemaValidity() const
{
CHIP_ERROR err = CHIP_NO_ERROR;
int tagPresenceMask = 0;
TLV::TLVReader reader;
PRETTY_PRINT("StatusIB =");
PRETTY_PRINT("{");
// make a copy of the reader
reader.Init(mReader);
while (CHIP_NO_ERROR == (err = reader.Next()))
{
if (!TLV::IsContextTag(reader.GetTag()))
{
continue;
}
if (!(tagPresenceMask & (1 << to_underlying(Tag::kStatus))))
{
tagPresenceMask |= (1 << to_underlying(Tag::kStatus));
#if CHIP_DETAIL_LOGGING
{
uint8_t status;
ReturnErrorOnFailure(reader.Get(status));
PRETTY_PRINT("\tstatus = " ChipLogFormatIMStatus ",", ChipLogValueIMStatus(static_cast<Status>(status)));
}
#endif // CHIP_DETAIL_LOGGING
}
else if (!(tagPresenceMask & (1 << to_underlying(Tag::kClusterStatus))))
{
tagPresenceMask |= (1 << to_underlying(Tag::kClusterStatus));
#if CHIP_DETAIL_LOGGING
{
ClusterStatus clusterStatus;
ReturnErrorOnFailure(reader.Get(clusterStatus));
PRETTY_PRINT("\tcluster-status = 0x%x,", clusterStatus);
}
#endif // CHIP_DETAIL_LOGGING
}
else
{
PRETTY_PRINT("\tExtra element in StatusIB");
}
}
PRETTY_PRINT("},");
PRETTY_PRINT("");
// if we have exhausted this container
if (CHIP_END_OF_TLV == err)
{
// check for required fields:
const int requiredFields = (1 << to_underlying(Tag::kStatus));
err = (tagPresenceMask & requiredFields) == requiredFields ? CHIP_NO_ERROR : CHIP_ERROR_IM_MALFORMED_STATUS_IB;
}
ReturnErrorOnFailure(err);
return reader.ExitContainer(mOuterContainerType);
}
#endif // CHIP_CONFIG_IM_ENABLE_SCHEMA_CHECK
StatusIB::Builder & StatusIB::Builder::EncodeStatusIB(const StatusIB & aStatusIB)
{
mError = mpWriter->Put(TLV::ContextTag(to_underlying(Tag::kStatus)), aStatusIB.mStatus);
SuccessOrExit(mError);
if (aStatusIB.mClusterStatus.HasValue())
{
mError = mpWriter->Put(TLV::ContextTag(to_underlying(Tag::kClusterStatus)), aStatusIB.mClusterStatus.Value());
SuccessOrExit(mError);
}
EndOfContainer();
exit:
return *this;
}
CHIP_ERROR StatusIB::ToChipError() const
{
if (mStatus == Status::Success)
{
return CHIP_NO_ERROR;
}
if (mClusterStatus.HasValue())
{
return ChipError(ChipError::SdkPart::kIMClusterStatus, mClusterStatus.Value());
}
return ChipError(ChipError::SdkPart::kIMGlobalStatus, to_underlying(mStatus));
}
void StatusIB::InitFromChipError(CHIP_ERROR aError)
{
if (aError.IsPart(ChipError::SdkPart::kIMClusterStatus))
{
mStatus = Status::Failure;
mClusterStatus = MakeOptional(aError.GetSdkCode());
return;
}
mClusterStatus = NullOptional;
if (aError == CHIP_NO_ERROR)
{
mStatus = Status::Success;
return;
}
if (aError.IsPart(ChipError::SdkPart::kIMGlobalStatus))
{
mStatus = static_cast<Status>(aError.GetSdkCode());
return;
}
mStatus = Status::Failure;
}
namespace {
bool FormatStatusIBError(char * buf, uint16_t bufSize, CHIP_ERROR err)
{
if (!err.IsIMStatus())
{
return false;
}
const char * desc = nullptr;
#if !CHIP_CONFIG_SHORT_ERROR_STR
constexpr char generalFormat[] = "General error: " ChipLogFormatIMStatus;
constexpr char clusterFormat[] = "Cluster-specific error: 0x%02x";
// Formatting an 8-bit int will take at most 2 chars, and replace the '%02x'
// so a buffer big enough to hold our format string will also hold our
// formatted string, as long as we account for the possible string formats.
constexpr size_t statusNameMaxLength =
#define CHIP_IM_STATUS_CODE(name, spec_name, value) \
max(sizeof(#spec_name),
#include <protocols/interaction_model/StatusCodeList.h>
#undef CHIP_IM_STATUS_CODE
static_cast<size_t>(0)
#define CHIP_IM_STATUS_CODE(name, spec_name, value) \
)
#include <protocols/interaction_model/StatusCodeList.h>
#undef CHIP_IM_STATUS_CODE
;
constexpr size_t formattedSize = max(sizeof(generalFormat) + statusNameMaxLength, sizeof(clusterFormat));
char formattedString[formattedSize];
StatusIB status;
status.InitFromChipError(err);
if (status.mClusterStatus.HasValue())
{
snprintf(formattedString, formattedSize, clusterFormat, status.mClusterStatus.Value());
}
else
{
snprintf(formattedString, formattedSize, generalFormat, ChipLogValueIMStatus(status.mStatus));
}
desc = formattedString;
#endif // !CHIP_CONFIG_SHORT_ERROR_STR
FormatError(buf, bufSize, "IM", err, desc);
return true;
}
} // anonymous namespace
void StatusIB::RegisterErrorFormatter()
{
static ErrorFormatter sStatusIBErrorFormatter = { FormatStatusIBError, nullptr };
::RegisterErrorFormatter(&sStatusIBErrorFormatter);
}
}; // namespace app
}; // namespace chip