From 06b3204f1b4e15083813df003b4ff45199af0e44 Mon Sep 17 00:00:00 2001 From: Zhanglong Xia Date: Tue, 10 Dec 2024 09:38:10 +0800 Subject: [PATCH] [spinel] save the radio spinel metrics to temporary settings The otRadioSpinelMetrics includes metrics that record the count of the RCP timeout and the count of RCP reboot and so on. After these events happen, the Thread stack will exits and the system will re-start the Thread stack again. In this case, metrics of the otRadioSpinelMetrics will be cleared and metrics values are awalys 0. This commit adds a temporary settings to store the radio spinel metrics. The temporary settings will clear all settings after the system reboots. --- src/lib/spinel/radio_spinel.cpp | 66 ++++++++- src/lib/spinel/radio_spinel.hpp | 70 ++++++++- src/posix/platform/CMakeLists.txt | 1 + src/posix/platform/openthread-posix-config.h | 10 ++ src/posix/platform/radio.cpp | 22 ++- src/posix/platform/radio.hpp | 32 ++++ src/posix/platform/temp_settings.cpp | 146 +++++++++++++++++++ src/posix/platform/temp_settings.hpp | 93 ++++++++++++ 8 files changed, 429 insertions(+), 11 deletions(-) create mode 100644 src/posix/platform/temp_settings.cpp create mode 100644 src/posix/platform/temp_settings.hpp diff --git a/src/lib/spinel/radio_spinel.cpp b/src/lib/spinel/radio_spinel.cpp index f467c1f2849..4311ede85ad 100644 --- a/src/lib/spinel/radio_spinel.cpp +++ b/src/lib/spinel/radio_spinel.cpp @@ -66,6 +66,7 @@ otRadioCaps RadioSpinel::sRadioCaps = OT_RADIO_CAPS_NONE; RadioSpinel::RadioSpinel(void) : Logger("RadioSpinel") , mInstance(nullptr) + , mCallbacks() , mCmdTidsInUse(0) , mCmdNextTid(1) , mTxRadioTid(0) @@ -108,6 +109,7 @@ RadioSpinel::RadioSpinel(void) , mTxRadioEndUs(UINT64_MAX) , mRadioTimeRecalcStart(UINT64_MAX) , mRadioTimeOffset(UINT64_MAX) + , mMetrics(mCallbacks) #if OPENTHREAD_SPINEL_CONFIG_VENDOR_HOOK_ENABLE , mVendorRestorePropertiesCallback(nullptr) , mVendorRestorePropertiesContext(nullptr) @@ -120,7 +122,6 @@ RadioSpinel::RadioSpinel(void) , mTimeSyncOn(false) , mSpinelDriver(nullptr) { - memset(&mRadioSpinelMetrics, 0, sizeof(mRadioSpinelMetrics)); memset(&mCallbacks, 0, sizeof(mCallbacks)); } @@ -170,6 +171,8 @@ void RadioSpinel::Init(bool aSkipRcpVersionCheck, mTxRadioFrame.mPsdu = mTxPsdu; mAckRadioFrame.mPsdu = mAckPsdu; + mMetrics.Init(); + exit: SuccessOrDie(error); } @@ -2005,7 +2008,7 @@ void RadioSpinel::HandleRcpUnexpectedReset(spinel_status_t aStatus) { OT_UNUSED_VARIABLE(aStatus); - mRadioSpinelMetrics.mRcpUnexpectedResetCount++; + mMetrics.IncrementRcpUnexpectedResetCount(); LogCrit("Unexpected RCP reset: %s", spinel_status_to_cstr(aStatus)); #if OPENTHREAD_SPINEL_CONFIG_RCP_RESTORATION_MAX_COUNT > 0 @@ -2019,7 +2022,7 @@ void RadioSpinel::HandleRcpUnexpectedReset(spinel_status_t aStatus) void RadioSpinel::HandleRcpTimeout(void) { - mRadioSpinelMetrics.mRcpTimeoutCount++; + mMetrics.IncrementRcpTimeoutCount(); #if OPENTHREAD_SPINEL_CONFIG_RCP_RESTORATION_MAX_COUNT > 0 mRcpFailure = kRcpFailureTimeout; @@ -2055,7 +2058,7 @@ void RadioSpinel::RecoverFromRcpFailure(void) LogWarn("RCP failure detected"); - ++mRadioSpinelMetrics.mRcpRestorationCount; + mMetrics.IncrementRcpRestorationCount(); ++mRcpFailureCount; if (mRcpFailureCount > kMaxFailureCount) { @@ -2457,5 +2460,60 @@ void RadioSpinel::HandleCompatibilityError(void) DieNow(OT_EXIT_RADIO_SPINEL_INCOMPATIBLE); } +void RadioSpinel::MetricsTracker::Init(void) { RestoreMetrics(); } + +void RadioSpinel::MetricsTracker::RestoreMetrics(void) +{ + VerifyOrExit((mCallbacks.mSaveRadioSpinelMetrics != nullptr) && (mCallbacks.mRestoreRadioSpinelMetrics != nullptr)); + + if (mCallbacks.mRestoreRadioSpinelMetrics(mMetrics, mCallbacks.mRadioSpinelMetricsContext) != OT_ERROR_NONE) + { + memset(&mMetrics, 0, sizeof(mMetrics)); + mCallbacks.mSaveRadioSpinelMetrics(mMetrics, mCallbacks.mRadioSpinelMetricsContext); + } + +exit: + return; +} + +void RadioSpinel::MetricsTracker::SaveMetrics(void) +{ + VerifyOrExit(mCallbacks.mSaveRadioSpinelMetrics != nullptr); + mCallbacks.mSaveRadioSpinelMetrics(mMetrics, mCallbacks.mRadioSpinelMetricsContext); + +exit: + return; +} + +void RadioSpinel::MetricsTracker::IncrementCount(MetricType aType) +{ + RestoreMetrics(); + + switch (aType) + { + case kTypeTimeoutCount: + mMetrics.mRcpTimeoutCount++; + break; + + case kTypeUnexpectResetCount: + mMetrics.mRcpUnexpectedResetCount++; + break; + + case kTypeRestorationCount: + mMetrics.mRcpRestorationCount++; + break; + + case kTypeSpinelParseErrorCount: + mMetrics.mSpinelParseErrorCount++; + break; + + default: + OT_ASSERT(false); + break; + } + + SaveMetrics(); +} + } // namespace Spinel } // namespace ot diff --git a/src/lib/spinel/radio_spinel.hpp b/src/lib/spinel/radio_spinel.hpp index 90de94ce7e2..38d1808bc2a 100644 --- a/src/lib/spinel/radio_spinel.hpp +++ b/src/lib/spinel/radio_spinel.hpp @@ -134,6 +134,28 @@ struct RadioSpinelCallbacks */ void (*mDiagTransmitDone)(otInstance *aInstance, otRadioFrame *aFrame, Error aError); #endif // OPENTHREAD_CONFIG_DIAG_ENABLE + + /** + * This method saves the radio spinel metrics to the temporary storage. + * + * @param[in] aMetrics A reference to the radio spinel metrics. + * @param[in] aContext A pointer to application-specific context. + */ + void (*mSaveRadioSpinelMetrics)(const otRadioSpinelMetrics &aMetrics, void *aContext); + + /** + * This method restores the radio spinel metrics from the temporary storage. + * + * @param[out] aMetrics A reference to the radio spinel metrics. + * @param[in] aContext A pointer to application-specific context. + */ + otError (*mRestoreRadioSpinelMetrics)(otRadioSpinelMetrics &aMetrics, void *aContext); + + /** + * The pointer to application-specific context for methods `mSaveRadioSpinelMetrics()` and + * `mRestoreRadioSpinelMetrics()`. + */ + void *mRadioSpinelMetricsContext; }; /** @@ -958,7 +980,7 @@ class RadioSpinel : private Logger * * @returns The radio Spinel metrics. */ - const otRadioSpinelMetrics *GetRadioSpinelMetrics(void) const { return &mRadioSpinelMetrics; } + const otRadioSpinelMetrics &GetRadioSpinelMetrics(void) const { return mMetrics.GetMetrics(); } #if OPENTHREAD_CONFIG_PLATFORM_POWER_CALIBRATION_ENABLE /** @@ -1198,7 +1220,10 @@ class RadioSpinel : private Logger void UpdateParseErrorCount(otError aError) { - mRadioSpinelMetrics.mSpinelParseErrorCount += (aError == OT_ERROR_PARSE) ? 1 : 0; + if (aError == OT_ERROR_PARSE) + { + mMetrics.IncrementSpinelParseErrorCount(); + } } otError SetMacKey(uint8_t aKeyIdMode, @@ -1216,6 +1241,42 @@ class RadioSpinel : private Logger void HandleCompatibilityError(void); + typedef otRadioSpinelMetrics Metrics; + + class MetricsTracker + { + public: + explicit MetricsTracker(RadioSpinelCallbacks &aCallbacks) + : mCallbacks(aCallbacks) + { + memset(&mMetrics, 0, sizeof(mMetrics)); + } + + void Init(void); + void IncrementRcpTimeoutCount(void) { IncrementCount(kTypeTimeoutCount); } + void IncrementRcpUnexpectedResetCount(void) { IncrementCount(kTypeUnexpectResetCount); } + void IncrementRcpRestorationCount(void) { IncrementCount(kTypeRestorationCount); } + void IncrementSpinelParseErrorCount(void) { IncrementCount(kTypeSpinelParseErrorCount); } + + const Metrics &GetMetrics(void) const { return mMetrics; } + + private: + enum MetricType : uint8_t + { + kTypeTimeoutCount, + kTypeUnexpectResetCount, + kTypeRestorationCount, + kTypeSpinelParseErrorCount, + }; + + void RestoreMetrics(void); + void SaveMetrics(void); + void IncrementCount(MetricType aType); + + RadioSpinelCallbacks &mCallbacks; + Metrics mMetrics; + }; + otInstance *mInstance; RadioSpinelCallbacks mCallbacks; ///< Callbacks for notifications of higher layer. @@ -1315,9 +1376,8 @@ class RadioSpinel : private Logger uint64_t mRadioTimeRecalcStart; ///< When to recalculate RCP time offset. uint64_t mRadioTimeOffset; ///< Time difference with estimated RCP time minus host time. - MaxPowerTable mMaxPowerTable; - - otRadioSpinelMetrics mRadioSpinelMetrics; + MaxPowerTable mMaxPowerTable; + MetricsTracker mMetrics; #if OPENTHREAD_SPINEL_CONFIG_VENDOR_HOOK_ENABLE otRadioSpinelVendorRestorePropertiesCallback mVendorRestorePropertiesCallback; diff --git a/src/posix/platform/CMakeLists.txt b/src/posix/platform/CMakeLists.txt index 4b94da29374..d572316a2ec 100644 --- a/src/posix/platform/CMakeLists.txt +++ b/src/posix/platform/CMakeLists.txt @@ -150,6 +150,7 @@ add_library(openthread-posix spinel_manager.cpp spi_interface.cpp system.cpp + temp_settings.cpp trel.cpp udp.cpp utils.cpp diff --git a/src/posix/platform/openthread-posix-config.h b/src/posix/platform/openthread-posix-config.h index a9849485c93..9aba6024ea5 100644 --- a/src/posix/platform/openthread-posix-config.h +++ b/src/posix/platform/openthread-posix-config.h @@ -425,4 +425,14 @@ #define OPENTHREAD_POSIX_CONFIG_TREL_SELECT_INFRA_IF 0 #endif +/** + * @def OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE + * + * Define as 1 to enable the temporary settings. The key-value pairs stored in the temporary settings will be erased + * after the system is rebooted. + */ +#ifndef OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE +#define OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE 1 +#endif + #endif // OPENTHREAD_PLATFORM_POSIX_CONFIG_H_ diff --git a/src/posix/platform/radio.cpp b/src/posix/platform/radio.cpp index 823af6784f7..41e503c603e 100644 --- a/src/posix/platform/radio.cpp +++ b/src/posix/platform/radio.cpp @@ -67,6 +67,9 @@ Radio::Radio(void) #if OPENTHREAD_POSIX_CONFIG_RCP_CAPS_DIAG_ENABLE , mRcpCapsDiag(mRadioSpinel) #endif +#if OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE + , mTempSettings() +#endif { } @@ -94,6 +97,12 @@ void Radio::Init(const char *aUrl) callbacks.mReceiveDone = otPlatRadioReceiveDone; callbacks.mTransmitDone = otPlatRadioTxDone; callbacks.mTxStarted = otPlatRadioTxStarted; +#if OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE + callbacks.mSaveRadioSpinelMetrics = SaveRadioSpinelMetrics; + callbacks.mRestoreRadioSpinelMetrics = RestoreRadioSpinelMetrics; + callbacks.mRadioSpinelMetricsContext = this; + mTempSettings.Init(); +#endif resetRadio = !mRadioUrl.HasParam("no-reset"); skipCompatibilityCheck = mRadioUrl.HasParam("skip-rcp-compatibility-check"); @@ -105,6 +114,15 @@ void Radio::Init(const char *aUrl) ProcessRadioUrl(mRadioUrl); } +void Radio::Deinit(void) +{ + mRadioSpinel.Deinit(); + +#if OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE + mTempSettings.Deinit(); +#endif +} + void Radio::ProcessRadioUrl(const RadioUrl &aRadioUrl) { const char *region; @@ -215,7 +233,7 @@ ot::Spinel::RadioSpinel &GetRadioSpinel(void) { return sRadio.GetRadioSpinel(); ot::Posix::RcpCapsDiag &GetRcpCapsDiag(void) { return sRadio.GetRcpCapsDiag(); } #endif -void platformRadioDeinit(void) { GetRadioSpinel().Deinit(); } +void platformRadioDeinit(void) { sRadio.Deinit(); } void platformRadioHandleStateChange(otInstance *aInstance, otChangedFlags aFlags) { @@ -1064,7 +1082,7 @@ otError otPlatResetToBootloader(otInstance *aInstance) } #endif -const otRadioSpinelMetrics *otSysGetRadioSpinelMetrics(void) { return GetRadioSpinel().GetRadioSpinelMetrics(); } +const otRadioSpinelMetrics *otSysGetRadioSpinelMetrics(void) { return &GetRadioSpinel().GetRadioSpinelMetrics(); } const otRcpInterfaceMetrics *otSysGetRcpInterfaceMetrics(void) { diff --git a/src/posix/platform/radio.hpp b/src/posix/platform/radio.hpp index b1b700fb7e7..7ae9120514e 100644 --- a/src/posix/platform/radio.hpp +++ b/src/posix/platform/radio.hpp @@ -35,6 +35,7 @@ #include "rcp_caps_diag.hpp" #include "spi_interface.hpp" #include "spinel_manager.hpp" +#include "temp_settings.hpp" #include "vendor_interface.hpp" #include "common/code_utils.hpp" #include "lib/spinel/radio_spinel.hpp" @@ -68,6 +69,11 @@ class Radio : public Logger */ void Init(const char *aUrl); + /** + * De-initializes the Thread radio. + */ + void Deinit(void); + /** * Acts as an accessor to the spinel interface instance used by the radio. * @@ -95,6 +101,28 @@ class Radio : public Logger void ProcessRadioUrl(const RadioUrl &aRadioUrl); void ProcessMaxPowerTable(const RadioUrl &aRadioUrl); +#if OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE + static void SaveRadioSpinelMetrics(const otRadioSpinelMetrics &aMetrics, void *aContext) + { + reinterpret_cast(aContext)->SaveRadioSpinelMetrics(aMetrics); + } + + void SaveRadioSpinelMetrics(const otRadioSpinelMetrics &aMetrics) + { + mTempSettings.SaveRadioSpinelMetrics(aMetrics); + } + + static otError RestoreRadioSpinelMetrics(otRadioSpinelMetrics &aMetrics, void *aContext) + { + return reinterpret_cast(aContext)->RestoreRadioSpinelMetrics(aMetrics); + } + + otError RestoreRadioSpinelMetrics(otRadioSpinelMetrics &aMetrics) + { + return mTempSettings.RestoreRadioSpinelMetrics(aMetrics); + } +#endif + #if OPENTHREAD_POSIX_CONFIG_SPINEL_HDLC_INTERFACE_ENABLE && OPENTHREAD_POSIX_CONFIG_SPINEL_SPI_INTERFACE_ENABLE static constexpr size_t kSpinelInterfaceRawSize = sizeof(ot::Posix::SpiInterface) > sizeof(ot::Posix::HdlcInterface) ? sizeof(ot::Posix::SpiInterface) @@ -125,6 +153,10 @@ class Radio : public Logger #if OPENTHREAD_POSIX_CONFIG_RCP_CAPS_DIAG_ENABLE RcpCapsDiag mRcpCapsDiag; #endif + +#if OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE + TempSettings mTempSettings; +#endif }; } // namespace Posix diff --git a/src/posix/platform/temp_settings.cpp b/src/posix/platform/temp_settings.cpp new file mode 100644 index 00000000000..8055f818090 --- /dev/null +++ b/src/posix/platform/temp_settings.cpp @@ -0,0 +1,146 @@ +/* + * Copyright (c) 2024, The OpenThread Authors. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * 3. Neither the name of the copyright holder nor the + * names of its contributors may be used to endorse or promote products + * derived from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +/** + * @file + * This file implements the temporary settings module for saving, restoring and deleting the temporarily settings. + * All temporary settings will be deleted after the system reboots. + */ + +#include "openthread-posix-config.h" +#include "platform-posix.h" + +#include +#include +#include +#ifdef __linux__ +#include +#else +#include +#endif +#include + +#include "temp_settings.hpp" +#include "common/code_utils.hpp" +#include "common/debug.hpp" +#include "common/encoding.hpp" + +#if OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE +namespace ot { +namespace Posix { + +void TempSettings::Init(void) +{ + otError error = OT_ERROR_NONE; + time_t storedBootTime; + time_t currentBootTime; + uint16_t valueLength = sizeof(time_t); + + VerifyOrDie(SettingsFileInit() == OT_ERROR_NONE, OT_EXIT_FAILURE); + + currentBootTime = GetBootTime(); + + error = mSettingsFile.Get(kKeyBootTime, 0, reinterpret_cast(&storedBootTime), &valueLength); + + // If failed to get the boot time or the current boot time doesn't match the stored boot time, it means + // the system has been rebooted. + if ((error != OT_ERROR_NONE) || (valueLength != sizeof(time_t)) || !BootTimeMatch(storedBootTime, currentBootTime)) + { + mSettingsFile.Wipe(); + mSettingsFile.Set(kKeyBootTime, reinterpret_cast(¤tBootTime), + static_cast(sizeof(currentBootTime))); + } +} + +void TempSettings::Deinit(void) { mSettingsFile.Deinit(); } + +void TempSettings::SaveRadioSpinelMetrics(const otRadioSpinelMetrics &aMetrics) +{ + mSettingsFile.Set(kKeyRadioSpinelMetrics, reinterpret_cast(&aMetrics), + static_cast(sizeof(aMetrics))); +} + +otError TempSettings::RestoreRadioSpinelMetrics(otRadioSpinelMetrics &aMetrics) +{ + uint16_t valueLength = sizeof(aMetrics); + + return mSettingsFile.Get(kKeyRadioSpinelMetrics, 0, reinterpret_cast(&aMetrics), &valueLength); +} + +otError TempSettings::SettingsFileInit(void) +{ + static constexpr size_t kMaxFileBaseNameSize = 32; + char fileBaseName[kMaxFileBaseNameSize]; + const char *offset = getenv("PORT_OFFSET"); + uint64_t eui64; + + otPlatRadioGetIeeeEui64(gInstance, reinterpret_cast(&eui64)); + eui64 = ot::BigEndian::HostSwap64(eui64); + + snprintf(fileBaseName, sizeof(fileBaseName), "%s_%" PRIx64 "-temp", ((offset == nullptr) ? "0" : offset), eui64); + VerifyOrDie(strlen(fileBaseName) < kMaxFileBaseNameSize, OT_EXIT_FAILURE); + + return mSettingsFile.Init(fileBaseName); +} + +time_t TempSettings::GetBootTime(void) +{ + time_t curTime; + time_t upTime; + +#ifdef __linux__ + struct sysinfo info; + VerifyOrDie(sysinfo(&info) == 0, OT_EXIT_ERROR_ERRNO); + upTime = info.uptime; +#else + { + int mib[] = {CTL_KERN, KERN_BOOTTIME}; + struct timeval boottime; + size_t size = sizeof(boottime); + + VerifyOrDie(sysctl(mib, OT_ARRAY_LENGTH(mib), &boottime, &size, NULL, 0) == 0, OT_EXIT_ERROR_ERRNO); + upTime = boottime.tv_sec; + } +#endif + + VerifyOrDie(time(&curTime) != -1, OT_EXIT_ERROR_ERRNO); + + return (curTime - upTime); +} + +bool TempSettings::BootTimeMatch(time_t aBootTimeA, time_t aBootTimeB) +{ + time_t diff = (aBootTimeA > aBootTimeB) ? (aBootTimeA - aBootTimeB) : (aBootTimeB - aBootTimeA); + + // The unit of the boot time is 1 second. The boot time has 1 second of jitter. + return diff < 2; +} + +} // namespace Posix +} // namespace ot +#endif // OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE diff --git a/src/posix/platform/temp_settings.hpp b/src/posix/platform/temp_settings.hpp new file mode 100644 index 00000000000..808ec4c1ede --- /dev/null +++ b/src/posix/platform/temp_settings.hpp @@ -0,0 +1,93 @@ +/* + * Copyright (c) 2024, The OpenThread Authors. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * 3. Neither the name of the copyright holder nor the + * names of its contributors may be used to endorse or promote products + * derived from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +#ifndef OT_POSIX_PLATFORM_STEMP_ETTINGS_HPP_ +#define OT_POSIX_PLATFORM_STEMP_ETTINGS_HPP_ + +#include "openthread-posix-config.h" +#include "platform-posix.h" + +#include "settings_file.hpp" + +#if OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE +namespace ot { +namespace Posix { + +class TempSettings +{ +public: + TempSettings(void) + : mSettingsFile() + { + } + + /** + * Performs the initialization for the temporary settings file. + */ + void Init(void); + + /** + * Performs the de-initialization for the temporary settings file. + */ + void Deinit(void); + + /** + * Saves the radio spinel metrics to the temporary settings file. + * + * @param[in] aMetrics A reference to the radio spinel metrics. + */ + void SaveRadioSpinelMetrics(const otRadioSpinelMetrics &aMetrics); + + /** + * Restores the radio spinel metrics from the temporary settings file. + * + * @param[out] aMetrics A reference to the radio spinel metrics. + * + * @retval OT_ERROR_NONE The radio spinel metrics was found and fetched successfully. + * @retval OT_ERROR_NOT_FOUND The radio spinel metrics was not found in the setting store. + */ + otError RestoreRadioSpinelMetrics(otRadioSpinelMetrics &aMetrics); + +private: + enum + { + kKeyBootTime = 1, + kKeyRadioSpinelMetrics = 2, + }; + + otError SettingsFileInit(void); + time_t GetBootTime(void); + bool BootTimeMatch(time_t aBootTimeA, time_t aBootTimeB); + + SettingsFile mSettingsFile; +}; + +} // namespace Posix +} // namespace ot +#endif // OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE +#endif // OT_POSIX_PLATFORM_STEMP_ETTINGS_HPP_