blob: 59f3a0df45314ba2e1eb5571532b518f2b18bb5b [file] [log] [blame]
/*
*
* Copyright (c) 2020 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.
*/
/**
* @file
* Provides an implementation of the PlatformManager object
* for Zephyr platforms.
*/
#if !defined(CONFIG_NORDIC_SECURITY_BACKEND)
#include <crypto/CHIPCryptoPAL.h> // nogncheck
#endif // !defined(CONFIG_NORDIC_SECURITY_BACKEND)
#include <platform/internal/CHIPDeviceLayerInternal.h>
#include <lib/support/SafeInt.h>
#include <lib/support/logging/CHIPLogging.h>
#include <platform/PlatformManager.h>
#include <platform/Zephyr/DiagnosticDataProviderImpl.h>
#include <platform/internal/GenericPlatformManagerImpl_Zephyr.ipp>
#include <malloc.h>
#include <zephyr/drivers/entropy.h>
namespace chip {
namespace DeviceLayer {
static K_THREAD_STACK_DEFINE(sChipThreadStack, CHIP_DEVICE_CONFIG_CHIP_TASK_STACK_SIZE);
PlatformManagerImpl PlatformManagerImpl::sInstance{ sChipThreadStack };
static k_timer sOperationalHoursSavingTimer;
#if !defined(CONFIG_NORDIC_SECURITY_BACKEND) && !defined(CONFIG_MBEDTLS_ZEPHYR_ENTROPY)
static bool sChipStackEntropySourceAdded = false;
static int app_entropy_source(void * data, unsigned char * output, size_t len, size_t * olen)
{
const struct device * entropy = DEVICE_DT_GET(DT_CHOSEN(zephyr_entropy));
uint16_t clampedLen;
if (CanCastTo<uint16_t>(len))
{
clampedLen = static_cast<uint16_t>(len);
}
else
{
clampedLen = UINT16_MAX;
}
int ret = entropy_get_entropy(entropy, output, clampedLen);
if (ret == 0)
{
*olen = clampedLen;
}
else
{
*olen = 0;
}
return ret;
}
#endif // !defined(CONFIG_NORDIC_SECURITY_BACKEND) && !defined(CONFIG_MBEDTLS_ZEPHYR_ENTROPY)
void PlatformManagerImpl::OperationalHoursSavingTimerEventHandler(k_timer * timer)
{
PlatformMgr().ScheduleWork(UpdateOperationalHours);
}
void PlatformManagerImpl::UpdateOperationalHours(intptr_t arg)
{
uint64_t upTimeS;
if (GetDiagnosticDataProvider().GetUpTime(upTimeS) != CHIP_NO_ERROR)
{
ChipLogError(DeviceLayer, "Failed to get up time of the node");
return;
}
uint64_t totalOperationalHours = 0;
const uint32_t upTimeH = upTimeS / 3600 < UINT32_MAX ? static_cast<uint32_t>(upTimeS / 3600) : UINT32_MAX;
const uint64_t deltaTime = upTimeH - sInstance.mSavedOperationalHoursSinceBoot;
if (ConfigurationMgr().GetTotalOperationalHours(reinterpret_cast<uint32_t &>(totalOperationalHours)) == CHIP_NO_ERROR)
{
ConfigurationMgr().StoreTotalOperationalHours(
static_cast<uint32_t>(totalOperationalHours + deltaTime < UINT32_MAX ? totalOperationalHours + deltaTime : UINT32_MAX));
sInstance.mSavedOperationalHoursSinceBoot = upTimeH;
}
else
{
ChipLogError(DeviceLayer, "Failed to get total operational hours of the node");
}
}
CHIP_ERROR PlatformManagerImpl::_InitChipStack(void)
{
CHIP_ERROR err;
#if !defined(CONFIG_NORDIC_SECURITY_BACKEND) && !defined(CONFIG_MBEDTLS_ZEPHYR_ENTROPY)
// Minimum required from source before entropy is released ( with mbedtls_entropy_func() ) (in bytes)
const size_t kThreshold = 16;
#endif // !defined(CONFIG_NORDIC_SECURITY_BACKEND) && !defined(CONFIG_MBEDTLS_ZEPHYR_ENTROPY)
// Initialize the configuration system.
err = Internal::ZephyrConfig::Init();
SuccessOrExit(err);
#if !defined(CONFIG_NORDIC_SECURITY_BACKEND) && !defined(CONFIG_MBEDTLS_ZEPHYR_ENTROPY)
if (!sChipStackEntropySourceAdded)
{
// Add entropy source based on Zephyr entropy driver
err = chip::Crypto::add_entropy_source(app_entropy_source, NULL, kThreshold);
SuccessOrExit(err);
sChipStackEntropySourceAdded = true;
}
#endif // !defined(CONFIG_NORDIC_SECURITY_BACKEND) && !defined(CONFIG_MBEDTLS_ZEPHYR_ENTROPY)
// Call _InitChipStack() on the generic implementation base class to finish the initialization process.
err = Internal::GenericPlatformManagerImpl_Zephyr<PlatformManagerImpl>::_InitChipStack();
SuccessOrExit(err);
// Start the timer to periodically save node operational hours.
k_timer_init(&sOperationalHoursSavingTimer, &PlatformManagerImpl::OperationalHoursSavingTimerEventHandler, nullptr);
k_timer_user_data_set(&sOperationalHoursSavingTimer, this);
k_timer_start(&sOperationalHoursSavingTimer, K_HOURS(CONFIG_CHIP_OPERATIONAL_TIME_SAVE_INTERVAL),
K_HOURS(CONFIG_CHIP_OPERATIONAL_TIME_SAVE_INTERVAL));
exit:
return err;
}
} // namespace DeviceLayer
} // namespace chip