Skip to content

Commit

Permalink
[spinel] save the radio spinel metrics to temporary settings
Browse files Browse the repository at this point in the history
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.
  • Loading branch information
zhanglongxia committed Jan 9, 2025
1 parent 3560870 commit 06b3204
Show file tree
Hide file tree
Showing 8 changed files with 429 additions and 11 deletions.
66 changes: 62 additions & 4 deletions src/lib/spinel/radio_spinel.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -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)
Expand Down Expand Up @@ -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)
Expand All @@ -120,7 +122,6 @@ RadioSpinel::RadioSpinel(void)
, mTimeSyncOn(false)
, mSpinelDriver(nullptr)
{
memset(&mRadioSpinelMetrics, 0, sizeof(mRadioSpinelMetrics));
memset(&mCallbacks, 0, sizeof(mCallbacks));
}

Expand Down Expand Up @@ -170,6 +171,8 @@ void RadioSpinel::Init(bool aSkipRcpVersionCheck,
mTxRadioFrame.mPsdu = mTxPsdu;
mAckRadioFrame.mPsdu = mAckPsdu;

mMetrics.Init();

exit:
SuccessOrDie(error);
}
Expand Down Expand Up @@ -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
Expand All @@ -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;
Expand Down Expand Up @@ -2055,7 +2058,7 @@ void RadioSpinel::RecoverFromRcpFailure(void)

LogWarn("RCP failure detected");

++mRadioSpinelMetrics.mRcpRestorationCount;
mMetrics.IncrementRcpRestorationCount();
++mRcpFailureCount;
if (mRcpFailureCount > kMaxFailureCount)
{
Expand Down Expand Up @@ -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
70 changes: 65 additions & 5 deletions src/lib/spinel/radio_spinel.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -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;
};

/**
Expand Down Expand Up @@ -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
/**
Expand Down Expand Up @@ -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,
Expand All @@ -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.
Expand Down Expand Up @@ -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;
Expand Down
1 change: 1 addition & 0 deletions src/posix/platform/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand Down
10 changes: 10 additions & 0 deletions src/posix/platform/openthread-posix-config.h
Original file line number Diff line number Diff line change
Expand Up @@ -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_
22 changes: 20 additions & 2 deletions src/posix/platform/radio.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -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
{
}

Expand Down Expand Up @@ -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");
Expand All @@ -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;
Expand Down Expand Up @@ -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)
{
Expand Down Expand Up @@ -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)
{
Expand Down
32 changes: 32 additions & 0 deletions src/posix/platform/radio.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -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"
Expand Down Expand Up @@ -68,6 +69,11 @@ class Radio : public Logger<Radio>
*/
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.
*
Expand Down Expand Up @@ -95,6 +101,28 @@ class Radio : public Logger<Radio>
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<Radio *>(aContext)->SaveRadioSpinelMetrics(aMetrics);
}

void SaveRadioSpinelMetrics(const otRadioSpinelMetrics &aMetrics)
{
mTempSettings.SaveRadioSpinelMetrics(aMetrics);
}

static otError RestoreRadioSpinelMetrics(otRadioSpinelMetrics &aMetrics, void *aContext)
{
return reinterpret_cast<Radio *>(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)
Expand Down Expand Up @@ -125,6 +153,10 @@ class Radio : public Logger<Radio>
#if OPENTHREAD_POSIX_CONFIG_RCP_CAPS_DIAG_ENABLE
RcpCapsDiag mRcpCapsDiag;
#endif

#if OPENTHREAD_POSIX_CONFIG_TEMP_SETTINGS_ENABLE
TempSettings mTempSettings;
#endif
};

} // namespace Posix
Expand Down
Loading

0 comments on commit 06b3204

Please sign in to comment.