blob: bac743fe5d0cc0514ce147f4884bd36926ecb20b [file] [log] [blame]
/*
*
* Copyright (c) 2023 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 "ICDConfigurationData.h"
#include <lib/support/CodeUtils.h>
namespace chip {
ICDConfigurationData ICDConfigurationData::instance;
System::Clock::Milliseconds32 ICDConfigurationData::GetSlowPollingInterval()
{
// When LIT capable device operates in SIT mode, it shall transition to use the mSITPollingInterval
// if this one is shorter than the configured mLITPollingInterval.
// Either way, the slow poll interval used SHALL NOT be greater than the SIT mode polling threshold, per spec.
// This is important for ICD device configured for LIT operation but currently operating as a SIT
// due to a lack of client registration
if (mFeatureMap.Has(app::Clusters::IcdManagement::Feature::kLongIdleTimeSupport) && mICDMode == ICDMode::SIT)
{
// mSITPollingInterval cannot be configured to a value greater than kSITPollingThreshold.
// The SIT slow polling interval compliance is therefore always respected by using the smallest
// value from mLITPollingInterval or mSITPollingInterval;
return std::min(mLITPollingInterval, mSITPollingInterval);
}
return mLITPollingInterval;
}
CHIP_ERROR ICDConfigurationData::SetSlowPollingInterval(System::Clock::Milliseconds32 slowPollInterval)
{
bool isLITSupported = mFeatureMap.Has(app::Clusters::IcdManagement::Feature::kLongIdleTimeSupport);
// If LIT is not supported, the slow polling interval cannot be set higher than kSITPollingThreshold.
VerifyOrReturnError((isLITSupported || slowPollInterval <= kSITPollingThreshold), CHIP_ERROR_INVALID_ARGUMENT);
mLITPollingInterval = slowPollInterval;
return CHIP_NO_ERROR;
};
CHIP_ERROR ICDConfigurationData::SetSITPollingInterval(System::Clock::Milliseconds32 pollingInterval)
{
VerifyOrReturnError(pollingInterval <= kSITPollingThreshold, CHIP_ERROR_INVALID_ARGUMENT);
mSITPollingInterval = pollingInterval;
return CHIP_NO_ERROR;
}
CHIP_ERROR ICDConfigurationData::SetModeDurations(Optional<System::Clock::Milliseconds32> activeModeDuration,
Optional<System::Clock::Milliseconds32> idleModeDuration)
{
VerifyOrReturnError(activeModeDuration.HasValue() || idleModeDuration.HasValue(), CHIP_ERROR_INVALID_ARGUMENT);
// Convert idleModeDuration to seconds for the correct precision
std::optional<System::Clock::Seconds32> tmpIdleModeDuration = std::nullopt;
if (idleModeDuration.HasValue())
tmpIdleModeDuration = std::chrono::duration_cast<System::Clock::Seconds32>(idleModeDuration.Value());
// Here we set shortIdleModeDuration the same as idleModeDuration to maintain the api previous behaviour
return SetModeDurations(activeModeDuration.std_optional(), tmpIdleModeDuration, tmpIdleModeDuration);
}
CHIP_ERROR ICDConfigurationData::SetModeDurations(std::optional<System::Clock::Milliseconds32> activeModeDuration,
std::optional<System::Clock::Seconds32> idleModeDuration,
std::optional<System::Clock::Seconds32> shortIdleModeDuration)
{
VerifyOrReturnError(activeModeDuration.has_value() || idleModeDuration.has_value() || shortIdleModeDuration.has_value(),
CHIP_ERROR_INVALID_ARGUMENT);
System::Clock::Milliseconds32 tmpActiveModeDuration = activeModeDuration.value_or(mActiveModeDuration);
System::Clock::Seconds32 tmpIdleModeDuration = idleModeDuration.value_or(mIdleModeDuration);
VerifyOrReturnError(tmpActiveModeDuration <= tmpIdleModeDuration, CHIP_ERROR_INVALID_ARGUMENT);
VerifyOrReturnError(tmpIdleModeDuration <= kMaxIdleModeDuration, CHIP_ERROR_INVALID_ARGUMENT);
VerifyOrReturnError(tmpIdleModeDuration >= kMinIdleModeDuration, CHIP_ERROR_INVALID_ARGUMENT);
System::Clock::Seconds32 tmpShortIdleModeDuration;
if (shortIdleModeDuration.has_value())
{
// shortIdleModeDuration was provided, it shall be lesser than or equal to idleModeDuration.
tmpShortIdleModeDuration = shortIdleModeDuration.value();
}
else
{
// shortIdleModeDuration was not provided. To ensure correct mode transitions and device compliance,
// shortIdleModeDuration must not exceed idleModeDuration, so we use the smaller of the current shortIdleModeDuration
// and the resultant idleModeDuration.
// This approach overwrites a previous valid shortIdleModeDuration rather than erroring the call but maintains the previous
// api behavior.
tmpShortIdleModeDuration = std::min(mShortIdleModeDuration, tmpIdleModeDuration);
}
VerifyOrReturnError(tmpShortIdleModeDuration <= tmpIdleModeDuration, CHIP_ERROR_INVALID_ARGUMENT);
mIdleModeDuration = tmpIdleModeDuration;
mActiveModeDuration = tmpActiveModeDuration;
mShortIdleModeDuration = tmpShortIdleModeDuration;
return CHIP_NO_ERROR;
}
bool ICDConfigurationData::ShouldUseShortIdle()
{
VerifyOrReturnValue(mShortIdleModeDuration < mIdleModeDuration, false);
return (mFeatureMap.Has(app::Clusters::IcdManagement::Feature::kLongIdleTimeSupport) && mICDMode == ICDMode::SIT);
}
System::Clock::Seconds32 ICDConfigurationData::GetModeBasedIdleModeDuration()
{
return ShouldUseShortIdle() ? mShortIdleModeDuration : mIdleModeDuration;
}
} // namespace chip