Skip to content

Commit

Permalink
[spinel] allow registering callback to handle compatibility errors
Browse files Browse the repository at this point in the history
  • Loading branch information
gytxxsy committed Oct 10, 2024
1 parent 287dbfa commit 58e2098
Show file tree
Hide file tree
Showing 3 changed files with 70 additions and 8 deletions.
9 changes: 9 additions & 0 deletions src/lib/spinel/openthread-spinel-config.h
Original file line number Diff line number Diff line change
Expand Up @@ -136,6 +136,15 @@
#define OPENTHREAD_SPINEL_CONFIG_VENDOR_HOOK_ENABLE 0
#endif

/**
* @def OPENTHREAD_SPINEL_CONFIG_COMPATIBILITY_ERROR_CALLBACK_ENABLE
*
* Enables compatibility error callback in Spinel
*/
#ifndef OPENTHREAD_SPINEL_CONFIG_COMPATIBILITY_ERROR_CALLBACK_ENABLE
#define OPENTHREAD_SPINEL_CONFIG_COMPATIBILITY_ERROR_CALLBACK_ENABLE 0
#endif

/**
* @def OPENTHREAD_SPINEL_CONFIG_VENDOR_HOOK_HEADER
*
Expand Down
35 changes: 29 additions & 6 deletions src/lib/spinel/radio_spinel.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -111,6 +111,10 @@ RadioSpinel::RadioSpinel(void)
#if OPENTHREAD_SPINEL_CONFIG_VENDOR_HOOK_ENABLE
, mVendorRestorePropertiesCallback(nullptr)
, mVendorRestorePropertiesContext(nullptr)
#endif
#if OPENTHREAD_SPINEL_CONFIG_COMPATIBILITY_ERROR_CALLBACK_ENABLE
, mCompatibilityErrorCallback(nullptr)
, mCompatibilityErrorContext(nullptr)
#endif
, mTimeSyncEnabled(false)
, mTimeSyncOn(false)
Expand Down Expand Up @@ -198,7 +202,7 @@ otError RadioSpinel::CheckSpinelVersion(void)
{
LogCrit("Spinel version mismatch - Posix:%d.%d, RCP:%d.%d", SPINEL_PROTOCOL_VERSION_THREAD_MAJOR,
SPINEL_PROTOCOL_VERSION_THREAD_MINOR, versionMajor, versionMinor);
DieNow(OT_EXIT_RADIO_SPINEL_INCOMPATIBLE);
HandleCompatibilityError();
}

exit:
Expand All @@ -210,13 +214,13 @@ void RadioSpinel::InitializeCaps(bool &aSupportsRcpApiVersion, bool &aSupportsRc
if (!GetSpinelDriver().CoprocessorHasCap(SPINEL_CAP_CONFIG_RADIO))
{
LogCrit("The co-processor isn't a RCP!");
DieNow(OT_EXIT_RADIO_SPINEL_INCOMPATIBLE);
HandleCompatibilityError();
}

if (!GetSpinelDriver().CoprocessorHasCap(SPINEL_CAP_MAC_RAW))
{
LogCrit("RCP capability list does not include support for radio/raw mode");
DieNow(OT_EXIT_RADIO_SPINEL_INCOMPATIBLE);
HandleCompatibilityError();
}

sSupportsLogStream = GetSpinelDriver().CoprocessorHasCap(SPINEL_CAP_OPENTHREAD_LOG_METADATA);
Expand Down Expand Up @@ -251,7 +255,7 @@ otError RadioSpinel::CheckRadioCapabilities(otRadioCaps aRequiredRadioCaps)
}
}

DieNow(OT_EXIT_RADIO_SPINEL_INCOMPATIBLE);
HandleCompatibilityError();
}

exit:
Expand Down Expand Up @@ -279,7 +283,7 @@ otError RadioSpinel::CheckRcpApiVersion(bool aSupportsRcpApiVersion, bool aSuppo
LogCrit("RCP and host are using incompatible API versions");
LogCrit("RCP API Version %u is older than min required by host %u", rcpApiVersion,
SPINEL_MIN_HOST_SUPPORTED_RCP_API_VERSION);
DieNow(OT_EXIT_RADIO_SPINEL_INCOMPATIBLE);
HandleCompatibilityError();
}
}

Expand All @@ -299,7 +303,7 @@ otError RadioSpinel::CheckRcpApiVersion(bool aSupportsRcpApiVersion, bool aSuppo
LogCrit("RCP and host are using incompatible API versions");
LogCrit("RCP requires min host API version %u but host is older and at version %u", minHostRcpApiVersion,
SPINEL_RCP_API_VERSION);
DieNow(OT_EXIT_RADIO_SPINEL_INCOMPATIBLE);
HandleCompatibilityError();
}
}

Expand Down Expand Up @@ -2395,5 +2399,24 @@ otError RadioSpinel::SetChannelTargetPower(uint8_t aChannel, int16_t aTargetPowe
}
#endif // OPENTHREAD_CONFIG_PLATFORM_POWER_CALIBRATION_ENABLE

#if OPENTHREAD_SPINEL_CONFIG_COMPATIBILITY_ERROR_CALLBACK_ENABLE
void RadioSpinel::SetCompatibilityErrorCallback(otRadioSpinelCompatibilityErrorCallback aCallback, void *aContext)
{
mCompatibilityErrorCallback = aCallback;
mCompatibilityErrorContext = aContext;
}
#endif

void RadioSpinel::HandleCompatibilityError(void)
{
#if OPENTHREAD_SPINEL_CONFIG_COMPATIBILITY_ERROR_CALLBACK_ENABLE
if (mCompatibilityErrorCallback)
{
mCompatibilityErrorCallback(mCompatibilityErrorContext);
}
#endif
DieNow(OT_EXIT_RADIO_SPINEL_INCOMPATIBLE);
}

} // namespace Spinel
} // namespace ot
34 changes: 32 additions & 2 deletions src/lib/spinel/radio_spinel.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -1006,8 +1006,10 @@ class RadioSpinel : private Logger

/**
* A callback type for restoring vendor properties.
*
* @param[in] aContext A pointer to the user context.
*/
typedef void (*otRadioSpinelVendorRestorePropertiesCallback)(void *context);
typedef void (*otRadioSpinelVendorRestorePropertiesCallback)(void *aContext);

/**
* Registers a callback to restore vendor properties.
Expand All @@ -1016,11 +1018,32 @@ class RadioSpinel : private Logger
* properties occurs (such as an unexpected RCP reset), the user can restore the vendor properties via the callback.
*
* @param[in] aCallback The callback.
* @param[in] aContext The context.
* @param[in] aContext A pointer to the user context.
*/
void SetVendorRestorePropertiesCallback(otRadioSpinelVendorRestorePropertiesCallback aCallback, void *aContext);
#endif // OPENTHREAD_SPINEL_CONFIG_VENDOR_HOOK_ENABLE

#if OPENTHREAD_SPINEL_CONFIG_COMPATIBILITY_ERROR_CALLBACK_ENABLE
/**
* A callback type for handling compatibility error of radio spinel.
*
* @param[in] aContext A pointer to the user context.
*/
typedef void (*otRadioSpinelCompatibilityErrorCallback)(void *aContext);

/**
* Registers a callback to handle error of radio spinel.
*
* This function is used to register a callback to handle radio spinel compatibility errors. When a radio spinel
* compatibility error occurs that cannot be resolved by a restart (e.g., RCP version mismatch), the user can
* handle the error through the callback(such as OTA) instead of letting the program crash directly.
*
* @param[in] aCallback The callback.
* @param[in] aContext A pointer to the user context.
*/
void SetCompatibilityErrorCallback(otRadioSpinelCompatibilityErrorCallback aCallback, void *aContext);
#endif

/**
* Enables or disables the time synchronization between the host and RCP.
*
Expand Down Expand Up @@ -1155,6 +1178,8 @@ class RadioSpinel : private Logger
void PlatDiagOutput(const char *aFormat, ...);
#endif

void HandleCompatibilityError(void);

otInstance *mInstance;

RadioSpinelCallbacks mCallbacks; ///< Callbacks for notifications of higher layer.
Expand Down Expand Up @@ -1262,6 +1287,11 @@ class RadioSpinel : private Logger
void *mVendorRestorePropertiesContext;
#endif

#if OPENTHREAD_SPINEL_CONFIG_COMPATIBILITY_ERROR_CALLBACK_ENABLE
otRadioSpinelCompatibilityErrorCallback mCompatibilityErrorCallback;
void *mCompatibilityErrorContext;
#endif

bool mTimeSyncEnabled : 1;
bool mTimeSyncOn : 1;

Expand Down

0 comments on commit 58e2098

Please sign in to comment.