blob: 69025b48f1fa06562dd9a624230b39db39c59245 [file] [log] [blame]
/*
* Copyright (c) 2023-2026 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 <app/clusters/temperature-control-server/TemperatureControlCluster.h>
#include <app/server-cluster/AttributeListBuilder.h>
#include <clusters/TemperatureControl/Metadata.h>
namespace chip::app::Clusters {
using namespace TemperatureControl;
using namespace TemperatureControl::Attributes;
// Specification defined ranges and limits
constexpr int16_t kMinTemperatureRange = -27315;
constexpr int16_t kMaxTemperatureRange = 32766;
constexpr int16_t kMinStep = 1;
constexpr uint8_t kMaxSelectedTemperatureLevel = 31;
constexpr uint8_t kMaxTemperatureLevelStringSize = 32;
TemperatureControl::SupportedTemperatureLevelsIteratorDelegate * TemperatureControlCluster::mDelegate = nullptr;
TemperatureControlCluster::TemperatureControlCluster(EndpointId endpointId, const BitFlags<Feature> features,
const StartupConfiguration & config) :
DefaultServerCluster({ endpointId, TemperatureControl::Id }),
mFeatures(features), mTemperatureSetpoint(config.temperatureSetpoint), mMinTemperature(config.minTemperature),
mMaxTemperature(config.maxTemperature), mStep(config.step), mSelectedTemperatureLevel(config.selectedTemperatureLevel)
{
if (mFeatures.Has(Feature::kTemperatureNumber))
{
VerifyOrDie(!mFeatures.Has(Feature::kTemperatureLevel));
VerifyOrDie(mMinTemperature >= kMinTemperatureRange && mMinTemperature <= kMaxTemperatureRange);
VerifyOrDie(mMaxTemperature >= mMinTemperature + 1);
VerifyOrDie(mTemperatureSetpoint >= mMinTemperature && mTemperatureSetpoint <= mMaxTemperature);
if (mFeatures.Has(Feature::kTemperatureStep))
{
VerifyOrDie(mStep >= kMinStep && mStep <= (mMaxTemperature - mMinTemperature));
}
}
if (mFeatures.Has(Feature::kTemperatureLevel))
{
VerifyOrDie(mSelectedTemperatureLevel <= kMaxSelectedTemperatureLevel);
}
}
DataModel::ActionReturnStatus TemperatureControlCluster::ReadAttribute(const DataModel::ReadAttributeRequest & request,
AttributeValueEncoder & encoder)
{
switch (request.path.mAttributeId)
{
case ClusterRevision::Id:
return encoder.Encode(TemperatureControl::kRevision);
case FeatureMap::Id:
return encoder.Encode(mFeatures);
case TemperatureSetpoint::Id:
return encoder.Encode(mTemperatureSetpoint);
case MinTemperature::Id:
return encoder.Encode(mMinTemperature);
case MaxTemperature::Id:
return encoder.Encode(mMaxTemperature);
case Step::Id:
return encoder.Encode(mStep);
case SelectedTemperatureLevel::Id:
return encoder.Encode(mSelectedTemperatureLevel);
case SupportedTemperatureLevels::Id:
if (mDelegate == nullptr)
{
return encoder.EncodeEmptyList();
}
mDelegate->Reset(request.path.mEndpointId);
return encoder.EncodeList([&](const auto & encod) -> CHIP_ERROR {
char buffer[kMaxTemperatureLevelStringSize] = { 0 };
MutableCharSpan item(buffer);
while (mDelegate->Next(item) == CHIP_NO_ERROR)
{
ReturnErrorOnFailure(encod.Encode(item));
item = MutableCharSpan(buffer);
}
return CHIP_NO_ERROR;
});
default:
return Protocols::InteractionModel::Status::UnsupportedAttribute;
}
}
CHIP_ERROR TemperatureControlCluster::Attributes(const ConcreteClusterPath & path,
ReadOnlyBufferBuilder<DataModel::AttributeEntry> & builder)
{
AttributeListBuilder listBuilder(builder);
const AttributeListBuilder::OptionalAttributeEntry optionalAttributes[] = {
{ mFeatures.Has(Feature::kTemperatureNumber), TemperatureSetpoint::kMetadataEntry },
{ mFeatures.Has(Feature::kTemperatureNumber), MinTemperature::kMetadataEntry },
{ mFeatures.Has(Feature::kTemperatureNumber), MaxTemperature::kMetadataEntry },
{ mFeatures.Has(Feature::kTemperatureStep), Step::kMetadataEntry },
{ mFeatures.Has(Feature::kTemperatureLevel), SelectedTemperatureLevel::kMetadataEntry },
{ mFeatures.Has(Feature::kTemperatureLevel), SupportedTemperatureLevels::kMetadataEntry },
};
return listBuilder.Append(Span(kMandatoryMetadata), Span(optionalAttributes));
}
CHIP_ERROR TemperatureControlCluster::SetTemperatureSetpoint(int16_t temperatureSetpoint)
{
VerifyOrReturnError(mFeatures.Has(Feature::kTemperatureNumber), CHIP_IM_GLOBAL_STATUS(InvalidInState));
VerifyOrReturnError(temperatureSetpoint >= mMinTemperature && temperatureSetpoint <= mMaxTemperature,
CHIP_IM_GLOBAL_STATUS(ConstraintError));
if (mFeatures.Has(Feature::kTemperatureStep))
{
VerifyOrReturnError((temperatureSetpoint - mMinTemperature) % mStep == 0, CHIP_IM_GLOBAL_STATUS(ConstraintError));
}
SetAttributeValue(mTemperatureSetpoint, temperatureSetpoint, TemperatureSetpoint::Id);
return CHIP_NO_ERROR;
}
CHIP_ERROR TemperatureControlCluster::SetSelectedTemperatureLevel(uint8_t selectedTemperatureLevel)
{
VerifyOrReturnError(mFeatures.Has(Feature::kTemperatureLevel), CHIP_IM_GLOBAL_STATUS(InvalidInState));
VerifyOrReturnError(mDelegate != nullptr, CHIP_IM_GLOBAL_STATUS(NotFound));
VerifyOrReturnError(selectedTemperatureLevel < mDelegate->Size(), CHIP_IM_GLOBAL_STATUS(ConstraintError));
mDelegate->Reset(mPath.mEndpointId);
SetAttributeValue(mSelectedTemperatureLevel, selectedTemperatureLevel, SelectedTemperatureLevel::Id);
return CHIP_NO_ERROR;
}
std::optional<DataModel::ActionReturnStatus> TemperatureControlCluster::InvokeCommand(const DataModel::InvokeRequest & request,
TLV::TLVReader & input_arguments,
CommandHandler * handler)
{
switch (request.path.mCommandId)
{
case Commands::SetTemperature::Id: {
Commands::SetTemperature::DecodableType data;
ReturnErrorOnFailure(data.Decode(input_arguments));
return HandleSetTemperature(handler, request.path, data);
}
default:
return Protocols::InteractionModel::Status::UnsupportedCommand;
}
}
CHIP_ERROR TemperatureControlCluster::AcceptedCommands(const ConcreteClusterPath & path,
ReadOnlyBufferBuilder<DataModel::AcceptedCommandEntry> & builder)
{
return builder.AppendElements({
Commands::SetTemperature::kMetadataEntry,
});
}
std::optional<DataModel::ActionReturnStatus>
TemperatureControlCluster::HandleSetTemperature(CommandHandler * commandObj, const ConcreteCommandPath & commandPath,
const TemperatureControl::Commands::SetTemperature::DecodableType & commandData)
{
auto & targetTemperature = commandData.targetTemperature;
auto & targetTemperatureLevel = commandData.targetTemperatureLevel;
using namespace chip::Protocols::InteractionModel;
if (mFeatures.Has(Feature::kTemperatureNumber))
{
VerifyOrReturnError(targetTemperature.HasValue(), Status::InvalidCommand);
CHIP_ERROR err = SetTemperatureSetpoint(targetTemperature.Value());
if (err == CHIP_IM_GLOBAL_STATUS(ConstraintError))
{
return Status::ConstraintError;
}
if (err != CHIP_NO_ERROR)
{
/**
* If the server is unable to execute the command at the time the command is received
* by the server (e.g. due to the design of a device it cannot accept a change in its
* temperature setting after it has begun operation), then the server SHALL respond
* with a status code of INVALID_IN_STATE, and discard the command.
**/
return Status::InvalidInState;
}
}
if (mFeatures.Has(Feature::kTemperatureLevel))
{
VerifyOrReturnError(targetTemperatureLevel.HasValue(), Status::InvalidCommand);
CHIP_ERROR err = SetSelectedTemperatureLevel(targetTemperatureLevel.Value());
if (err == CHIP_IM_GLOBAL_STATUS(NotFound))
{
return Status::NotFound;
}
if (err == CHIP_IM_GLOBAL_STATUS(ConstraintError))
{
return Status::ConstraintError;
}
if (err != CHIP_NO_ERROR)
{
/**
* If the server is unable to execute the command at the time the command is received
* by the server (e.g. due to the design of a device it cannot accept a change in its
* temperature setting after it has begun operation), then the server SHALL respond
* with a status code of INVALID_IN_STATE, and discard the command.
**/
return Status::InvalidInState;
}
}
return Status::Success;
}
} // namespace chip::app::Clusters