From 723a4f095250b75289ea472447a5c02ffcf1e212 Mon Sep 17 00:00:00 2001 From: Mason Tran Date: Mon, 23 Oct 2023 17:20:38 -0400 Subject: [PATCH] [sdk] update to Gecko SDK 4.3.2 --- src/src/diag.c | 90 +++++++- src/src/openthread-core-efr32-config.h | 67 +++++- src/src/radio.c | 280 ++++++++++++++----------- src/src/radio_extension.c | 2 +- src/src/system.c | 15 +- third_party/silabs/gecko_sdk | 2 +- 6 files changed, 312 insertions(+), 144 deletions(-) diff --git a/src/src/diag.c b/src/src/diag.c index 02d18753..12dd636f 100644 --- a/src/src/diag.c +++ b/src/src/diag.c @@ -32,7 +32,6 @@ * */ -#include #include #include #include @@ -47,6 +46,7 @@ #include "common/logging.hpp" #include "diag.h" +#include "em_gpio.h" #include "platform-band.h" #include "platform-efr32.h" #include "rail_ieee802154.h" @@ -63,6 +63,11 @@ #include "sl_rail_util_ant_div.h" #endif +#define GPIO_PIN_BITMASK 0xFFFFUL +#define GPIO_PORT_BITMASK (0xFFFFUL << 16) +#define GET_GPIO_PIN(x) (x & GPIO_PIN_BITMASK) +#define GET_GPIO_PORT(x) ((x & GPIO_PORT_BITMASK) >> 16) + struct PlatformDiagCommand { const char *mName; @@ -312,4 +317,87 @@ void otPlatDiagAlarmCallback(otInstance *aInstance) OT_UNUSED_VARIABLE(aInstance); } +static otError getGpioPortAndPin(uint32_t aGpio, uint16_t *aPort, uint16_t *aPin) +{ + otError error = OT_ERROR_NONE; + *aPort = GET_GPIO_PORT(aGpio); + *aPin = GET_GPIO_PIN(aGpio); + + if (*aPort > GPIO_PORT_MAX || *aPin > GPIO_PIN_MAX) + { + ExitNow(error = OT_ERROR_INVALID_ARGS); + } + +exit: + return error; +} + +otError otPlatDiagGpioSet(uint32_t aGpio, bool aValue) +{ + otError error; + uint16_t port; + uint16_t pin; + + SuccessOrExit(error = getGpioPortAndPin(aGpio, &port, &pin)); + + if (aValue) + { + GPIO_PinOutSet((GPIO_Port_TypeDef)port, pin); + } + else + { + GPIO_PinOutClear((GPIO_Port_TypeDef)port, pin); + } + +exit: + return error; +} + +otError otPlatDiagGpioGet(uint32_t aGpio, bool *aValue) +{ + otError error; + uint16_t port; + uint16_t pin; + + SuccessOrExit(error = getGpioPortAndPin(aGpio, &port, &pin)); + + *aValue = GPIO_PinInGet((GPIO_Port_TypeDef)port, pin); + +exit: + return error; +} + +otError otPlatDiagGpioSetMode(uint32_t aGpio, otGpioMode aMode) +{ + otError error; + uint16_t port; + uint16_t pin; + GPIO_Mode_TypeDef mode; + + SuccessOrExit(error = getGpioPortAndPin(aGpio, &port, &pin)); + + mode = (aMode == OT_GPIO_MODE_INPUT) ? gpioModeInput : gpioModePushPull; + + GPIO_PinModeSet((GPIO_Port_TypeDef)port, pin, mode, 0 /*out*/); + +exit: + return error; +} + +otError otPlatDiagGpioGetMode(uint32_t aGpio, otGpioMode *aMode) +{ + otError error; + uint16_t port; + uint16_t pin; + GPIO_Mode_TypeDef mode; + + SuccessOrExit(error = getGpioPortAndPin(aGpio, &port, &pin)); + + mode = GPIO_PinModeGet((GPIO_Port_TypeDef)port, pin); + + *aMode = (mode == gpioModeInput) ? OT_GPIO_MODE_INPUT : OT_GPIO_MODE_OUTPUT; + +exit: + return error; +} #endif // OPENTHREAD_CONFIG_DIAG_ENABLE diff --git a/src/src/openthread-core-efr32-config.h b/src/src/openthread-core-efr32-config.h index c653efc3..e0079ecf 100644 --- a/src/src/openthread-core-efr32-config.h +++ b/src/src/openthread-core-efr32-config.h @@ -35,6 +35,14 @@ #ifndef OPENTHREAD_CORE_EFR32_CONFIG_H_ #define OPENTHREAD_CORE_EFR32_CONFIG_H_ +#include "sl_device_init_hfxo.h" +#include "sl_device_init_hfxo_config.h" + +#if defined(HARDWARE_BOARD_HAS_LFXO) +#include "sl_device_init_lfxo.h" +#include "sl_device_init_lfxo_config.h" +#endif + // Use (user defined) application config file to define OpenThread configurations #ifdef SL_OPENTHREAD_APPLICATION_CONFIG_FILE #include SL_OPENTHREAD_APPLICATION_CONFIG_FILE @@ -200,9 +208,23 @@ * The minimum time (in microseconds) before the MHR start that the radio should be in receive state and ready to * properly receive in order to properly receive any IEEE 802.15.4 frame. Defaults to the duration of SHR + PHR. * + * 802.15.4 2.4GHz OQPSK: + * SHR: 4 bytes of preamble, 1 byte of sync word + * PHR: 1 byte + * Total (6 * 32) = 192 us. + * + * Proprietary SubGhz (2GFSK in 915MHz): + * SHR: 4 bytes preamble, 2 bytes SFD = 6 bytes + * PHR: 2 bytes + * Total (8 * 32) = 256 us. + * */ #ifndef OPENTHREAD_CONFIG_MIN_RECEIVE_ON_AHEAD -#define OPENTHREAD_CONFIG_MIN_RECEIVE_ON_AHEAD (6 * 32) +#if RADIO_CONFIG_SUBGHZ_SUPPORT +#define OPENTHREAD_CONFIG_MIN_RECEIVE_ON_AHEAD 256 +#else +#define OPENTHREAD_CONFIG_MIN_RECEIVE_ON_AHEAD 192 +#endif #endif /** @@ -213,9 +235,12 @@ * plus the duration of maximum enh-ack frame. Platforms are encouraged to improve this value for energy * efficiency purposes. * + * In theory, RAIL should automatically extend the duration of the receive window once the SHR has been + * detected, so we should be able to set this to zero. + * */ #ifndef OPENTHREAD_CONFIG_MIN_RECEIVE_ON_AFTER -#define OPENTHREAD_CONFIG_MIN_RECEIVE_ON_AFTER ((127 + 6 + 39) * 32) +#define OPENTHREAD_CONFIG_MIN_RECEIVE_ON_AFTER 0 #endif /* @@ -423,34 +448,54 @@ * */ #ifndef SL_OPENTHREAD_CSL_TX_UNCERTAINTY +#if OPENTHREAD_RADIO || OPENTHREAD_CONFIG_REFERENCE_DEVICE_ENABLE #define SL_OPENTHREAD_CSL_TX_UNCERTAINTY 175 +#elif OPENTHREAD_FTD +// Approx. ~128 us. for single CCA + some additional tx uncertainty in testing +#define SL_OPENTHREAD_CSL_TX_UNCERTAINTY 20 +#else +// Approx. ~128 us. for single CCA +// +// Note: Our SSEDs "schedule" transmissions to their parent in order to know +// exactly when in the future the data packets go out so they can calculate +// the accurate CSL phase to send to their parent. +// +// The receive windows on the SSEDs scale with this value, so increasing this +// uncertainty to account for full CCA/CSMA with 0..7 backoffs +// (see RAIL_CSMA_CONFIG_802_15_4_2003_2p4_GHz_OQPSK_CSMA) will mean that the +// receive windows can get very long (~ 5ms.) +// +// We have updated SSEDs to use a single CCA (RAIL_CSMA_CONFIG_SINGLE_CCA) +// instead. If they are in very busy channels, CSL won't be reliable anyway. +#define SL_OPENTHREAD_CSL_TX_UNCERTAINTY 12 +#endif #endif /** * @def SL_OPENTHREAD_HFXO_ACCURACY * - * XTAL accuracy in units of ± ppm. Also used for calculations during CSL operations. - * - * According to EFR datasheets, HFXO is ± 40 ppm. + * Worst case XTAL accuracy in units of ± ppm. Also used for calculations during CSL operations. * * @note Platforms may optimize this value based on operational conditions (i.e.: temperature). * */ #ifndef SL_OPENTHREAD_HFXO_ACCURACY -#define SL_OPENTHREAD_HFXO_ACCURACY 80 +#define SL_OPENTHREAD_HFXO_ACCURACY SL_DEVICE_INIT_HFXO_PRECISION #endif /** * @def SL_OPENTHREAD_LFXO_ACCURACY * - * XTAL accuracy in units of ± ppm. Also used for calculations during CSL operations. - * - * According to EFR datasheets, LFXO (at least for MG12) is -8 to +40 ppm. + * Worst case XTAL accuracy in units of ± ppm. Also used for calculations during CSL operations. * * @note Platforms may optimize this value based on operational conditions (i.e.: temperature). */ #ifndef SL_OPENTHREAD_LFXO_ACCURACY -#define SL_OPENTHREAD_LFXO_ACCURACY 48 +#if defined(HARDWARE_BOARD_HAS_LFXO) +#define SL_OPENTHREAD_LFXO_ACCURACY SL_DEVICE_INIT_LFXO_PRECISION +#else +#define SL_OPENTHREAD_LFXO_ACCURACY 0 +#endif #endif /** @@ -470,7 +515,7 @@ * */ #ifndef SL_OPENTHREAD_RADIO_RX_BUFFER_COUNT -#define SL_OPENTHREAD_RADIO_RX_BUFFER_COUNT 4 +#define SL_OPENTHREAD_RADIO_RX_BUFFER_COUNT 16 #endif /** diff --git a/src/src/radio.c b/src/src/radio.c index 16be0af9..86c95c04 100644 --- a/src/src/radio.c +++ b/src/src/radio.c @@ -88,6 +88,11 @@ #ifdef SL_CATALOG_OT_RCP_GP_INTERFACE_PRESENT #include "sl_rcp_gp_interface.h" #endif + +#ifdef SL_CATALOG_RAIL_UTIL_IEEE802154_FAST_CHANNEL_SWITCHING_PRESENT +#include "sl_rail_util_ieee802154_fast_channel_switching_config.h" +#endif // SL_CATALOG_RAIL_UTIL_IEEE802154_FAST_CHANNEL_SWITCHING_PRESENT + //------------------------------------------------------------------------------ // Enums, macros and static variables @@ -173,6 +178,19 @@ #define DEVICE_CAPABILITY_MCU_EN (DEVINFO->SWCAPA1 & _DEVINFO_SWCAPA1_RFMCUEN_MASK) #endif +static otRadioCaps sRadioCapabilities = + (OT_RADIO_CAPS_ACK_TIMEOUT | OT_RADIO_CAPS_CSMA_BACKOFF | OT_RADIO_CAPS_ENERGY_SCAN | OT_RADIO_CAPS_SLEEP_TO_TX +#if (OPENTHREAD_CONFIG_THREAD_VERSION >= OT_THREAD_VERSION_1_2) + | OT_RADIO_CAPS_TRANSMIT_SEC + // When scheduled tx is required, we support RAIL_StartScheduledCcaCsmaTx + // (delay is indicated in tx frame info set in MAC) + | OT_RADIO_CAPS_TRANSMIT_TIMING + // When scheduled rx is required, we support RAIL_ScheduleRx in our + // implementation of otPlatRadioReceiveAt + | OT_RADIO_CAPS_RECEIVE_TIMING +#endif + ); + // Energy Scan typedef enum { @@ -256,6 +274,10 @@ static otExtAddress sExtAddress[RADIO_EXT_ADDR_COUNT]; static int8_t sMaxChannelPower[RADIO_INTERFACE_COUNT][SL_MAX_CHANNELS_SUPPORTED]; static int8_t sDefaultTxPower[RADIO_INTERFACE_COUNT]; +// CSMA config: Should be globally scoped +RAIL_CsmaConfig_t csmaConfig = RAIL_CSMA_CONFIG_802_15_4_2003_2p4_GHz_OQPSK_CSMA; +RAIL_CsmaConfig_t cslCsmaConfig = RAIL_CSMA_CONFIG_SINGLE_CCA; + #if OPENTHREAD_CONFIG_MAC_HEADER_IE_SUPPORT static otRadioIeInfo sTransmitIeInfo; #endif @@ -279,7 +301,7 @@ efr32RadioCounters railDebugCounters; extern uint8_t otNcpPlatGetCurCommandIid(void); static uint8_t sRailFilterMask = RADIO_BCAST_PANID_FILTER_MASK; -#ifdef SL_CATALOG_RAIL_UTIL_IEEE802154_FAST_CHANNEL_SWITCHING_PRESENT +#if SL_RAIL_UTIL_IEEE802154_FAST_CHANNEL_SWITCHING_ENABLED #define FAST_CHANNEL_SWITCHING_SUPPORT 1 static RAIL_IEEE802154_RxChannelSwitchingCfg_t sChannelSwitchingCfg; @@ -320,7 +342,7 @@ static uint8_t fastChannelIndex(uint8_t aChannel) return INVALID_VALUE; } -#endif // SL_CATALOG_RAIL_UTIL_IEEE802154_FAST_CHANNEL_SWITCHING_PRESENT +#endif // SL_RAIL_UTIL_IEEE802154_FAST_CHANNEL_SWITCHING_ENABLED #else // OPENTHREAD_RADIO && OPENTHREAD_CONFIG_MULTIPAN_RCP_ENABLE == 1 #define otNcpPlatGetCurCommandIid() 0 @@ -337,22 +359,6 @@ static void sl_ot_coex_counter_on_event(sl_rail_util_coex_event_t event); RAIL_Handle_t emPhyRailHandle; #endif // SL_CATALOG_RAIL_MULTIPLEXER_PRESENT -#ifdef NONCOMPLIANT_ACK_TIMING_WORKAROUND -static RAIL_StateTiming_t gTimings = { - .idleToRx = 100, - .txToRx = 192 - 10, - .idleToTx = 100, -#if OPENTHREAD_CONFIG_THREAD_VERSION >= OT_THREAD_VERSION_1_2 - .rxToTx = 256, // accommodate enhanced ACKs -#else - .rxToTx = 192, -#endif - .rxSearchTimeout = 0, - .txToRxSearchTimeout = 0, - .txToTx = 0, -}; -#endif - static const RAIL_IEEE802154_Config_t sRailIeee802154Config = { .addresses = NULL, .ackConfig = @@ -392,8 +398,13 @@ static const RAIL_IEEE802154_Config_t sRailIeee802154Config = { #if RADIO_CONFIG_SUBGHZ_SUPPORT #define PHY_HEADER_SIZE 2 +// SHR: 4 bytes preamble, 2 bytes SFD +// 802.15.4 spec describes GFSK SHR to be the same format as SUN O-QPSK +// except preamble is 32 symbols (4 octets). +#define SHR_SIZE 6 #else #define PHY_HEADER_SIZE 1 +#define SHR_SIZE 5 // 4 bytes of preamble, 1 byte sync-word #endif // Misc @@ -646,11 +657,17 @@ static securityMaterial sMacKeys[RADIO_INTERFACE_COUNT]; static uint32_t sCslPeriod; static uint32_t sCslSampleTime; -static uint16_t getCslPhase() +static uint16_t getCslPhase(uint32_t shrTxTime) { - uint32_t curTime = otPlatAlarmMicroGetNow(); uint32_t cslPeriodInUs = sCslPeriod * OT_US_PER_TEN_SYMBOLS; - uint32_t diff = ((sCslSampleTime % cslPeriodInUs) - (curTime % cslPeriodInUs) + cslPeriodInUs) % cslPeriodInUs; + uint32_t diff; + + if (shrTxTime == 0U) + { + shrTxTime = otPlatAlarmMicroGetNow(); + } + + diff = ((sCslSampleTime % cslPeriodInUs) - (shrTxTime % cslPeriodInUs) + cslPeriodInUs) % cslPeriodInUs; return (uint16_t)(diff / OT_US_PER_TEN_SYMBOLS); } @@ -728,20 +745,6 @@ static otError radioProcessTransmitSecurity(otRadioFrame *aFrame, uint8_t iid) aFrame->mInfo.mTxInfo.mAesKey = &sMacKeys[iid].keys[keyToUse]; -#if defined(_SILICON_LABS_32B_SERIES_2) && (OPENTHREAD_CONFIG_CRYPTO_LIB == OPENTHREAD_CONFIG_CRYPTO_LIB_PSA) - otMacKeyMaterial aesKey; - size_t aKeyLen; - - otEXPECT_ACTION(otPlatCryptoExportKey(sMacKeys[iid].keys[keyToUse].mKeyMaterial.mKeyRef, - aesKey.mKeyMaterial.mKey.m8, - sizeof(aesKey.mKeyMaterial.mKey.m8), - &aKeyLen) - == OT_ERROR_NONE, - error = OT_ERROR_SECURITY); - - aFrame->mInfo.mTxInfo.mAesKey = &aesKey; -#endif - if (!aFrame->mInfo.mTxInfo.mIsHeaderUpdated) { if (otMacFrameIsAck(aFrame)) @@ -817,7 +820,7 @@ static void RAILCb_Generic(RAIL_Handle_t aRailHandle, RAIL_Events_t aEvents); static void efr32PhyStackInit(void); #if OPENTHREAD_CONFIG_THREAD_VERSION >= OT_THREAD_VERSION_1_2 -static void updateIeInfoTxFrame(void); +static void updateIeInfoTxFrame(uint32_t shrTxTime); #endif #ifdef SL_CATALOG_RAIL_UTIL_COEX_PRESENT @@ -1009,8 +1012,7 @@ static otError radioScheduleRx(uint8_t aChannel, uint32_t aStart, uint32_t aDura .endMode = RAIL_TIME_DELAY, .rxTransitionEndSchedule = 1, // This lets us idle after a scheduled-rx .hardWindowEnd = 0}; // This lets us receive a packet near a window-end-event - - status = RAIL_ScheduleRx(gRailHandle, aChannel, &rxCfg, &bgRxSchedulerInfo); + status = RAIL_ScheduleRx(gRailHandle, aChannel, &rxCfg, &bgRxSchedulerInfo); otEXPECT_ACTION(status == RAIL_STATUS_NO_ERROR, error = OT_ERROR_FAILED); (void)handlePhyStackEvent(SL_RAIL_UTIL_IEEE802154_STACK_EVENT_RX_LISTEN, 0U); @@ -1618,21 +1620,29 @@ otError otPlatRadioTransmit(otInstance *aInstance, otRadioFrame *aFrame) setInternalFlag(RADIO_TX_EVENTS, false); sTxFrame = aFrame; -#if OPENTHREAD_CONFIG_THREAD_VERSION >= OT_THREAD_VERSION_1_2 - uint8_t iid = 0; -#endif - -#if OPENTHREAD_RADIO && OPENTHREAD_CONFIG_MULTIPAN_RCP_ENABLE == 1 - iid = aFrame->mIid; -#endif - setInternalFlag(FLAG_CURRENT_TX_USE_CSMA, aFrame->mInfo.mTxInfo.mCsmaCaEnabled); #if OPENTHREAD_CONFIG_THREAD_VERSION >= OT_THREAD_VERSION_1_2 - updateIeInfoTxFrame(); +#if OPENTHREAD_CONFIG_MAC_CSL_RECEIVER_ENABLE + if (sCslPeriod > 0 && sTxFrame->mInfo.mTxInfo.mTxDelay == 0) + { + // Only called for CSL children (sCslPeriod > 0) + // Note: Our SSEDs "schedule" transmissions to their parent in order to know + // exactly when in the future the data packets go out so they can calculate + // the accurate CSL phase to send to their parent. + sTxFrame->mInfo.mTxInfo.mTxDelayBaseTime = RAIL_GetTime(); + sTxFrame->mInfo.mTxInfo.mTxDelay = 3000; // Chosen after internal certification testing + } +#endif + updateIeInfoTxFrame(sTxFrame->mInfo.mTxInfo.mTxDelayBaseTime + sTxFrame->mInfo.mTxInfo.mTxDelay + 160); // Note - we need to call this outside of txCurrentPacket as for Series 2, // this results in calling the SE interface from a critical section which is not permitted. - otEXPECT_ACTION(radioProcessTransmitSecurity(sTxFrame, iid) == OT_ERROR_NONE, error = OT_ERROR_INVALID_STATE); + +#if OPENTHREAD_RADIO && OPENTHREAD_CONFIG_MULTIPAN_RCP_ENABLE == 1 + radioProcessTransmitSecurity(sTxFrame, sTxFrame->mIid); +#else + radioProcessTransmitSecurity(sTxFrame, 0); +#endif #endif // OPENTHREAD_CONFIG_THREAD_VERSION >= OT_THREAD_VERSION_1_2 CORE_DECLARE_IRQ_STATE; @@ -1652,7 +1662,7 @@ otError otPlatRadioTransmit(otInstance *aInstance, otRadioFrame *aFrame) } #if OPENTHREAD_CONFIG_THREAD_VERSION >= OT_THREAD_VERSION_1_2 -void updateIeInfoTxFrame(void) +void updateIeInfoTxFrame(uint32_t shrTxTime) { assert(sTxFrame != NULL); @@ -1678,8 +1688,10 @@ void updateIeInfoTxFrame(void) // Update IE data in the 802.15.4 header with the newest CSL period / phase if (sCslPeriod > 0 && !sTxFrame->mInfo.mTxInfo.mIsHeaderUpdated) { - otMacFrameSetCslIe(sTxFrame, (uint16_t)sCslPeriod, getCslPhase()); + otMacFrameSetCslIe(sTxFrame, (uint16_t)sCslPeriod, getCslPhase(shrTxTime)); } +#else + OT_UNUSED_VARIABLE(shrTxTime); #endif // OPENTHREAD_CONFIG_MAC_CSL_RECEIVER_ENABLE } #endif // OPENTHREAD_CONFIG_THREAD_VERSION >= OT_THREAD_VERSION_1_2 @@ -1689,11 +1701,10 @@ void txCurrentPacket(void) assert(getInternalFlag(FLAG_ONGOING_TX_DATA)); assert(sTxFrame != NULL); - RAIL_CsmaConfig_t csmaConfig = RAIL_CSMA_CONFIG_802_15_4_2003_2p4_GHz_OQPSK_CSMA; - RAIL_TxOptions_t txOptions = RAIL_TX_OPTIONS_DEFAULT; - RAIL_Status_t status = RAIL_STATUS_INVALID_STATE; - uint8_t frameLength; - bool ackRequested; + RAIL_TxOptions_t txOptions = RAIL_TX_OPTIONS_DEFAULT; + RAIL_Status_t status = RAIL_STATUS_INVALID_STATE; + uint8_t frameLength; + bool ackRequested; #if RADIO_CONFIG_DEBUG_COUNTERS_SUPPORT railDebugCounters.mRailPlatTxTriggered++; @@ -1774,55 +1785,67 @@ void txCurrentPacket(void) } #endif - if (getInternalFlag(FLAG_CURRENT_TX_USE_CSMA)) + if (sTxFrame->mInfo.mTxInfo.mTxDelay == 0) { + if (getInternalFlag(FLAG_CURRENT_TX_USE_CSMA)) + { #if RADIO_CONFIG_DMP_SUPPORT - // time needed for CSMA/CA - txSchedulerInfo.transactionTime += RADIO_TIMING_CSMA_OVERHEAD_US; + // time needed for CSMA/CA + txSchedulerInfo.transactionTime += RADIO_TIMING_CSMA_OVERHEAD_US; #endif - csmaConfig.csmaTries = sTxFrame->mInfo.mTxInfo.mMaxCsmaBackoffs; - csmaConfig.ccaThreshold = sCcaThresholdDbm; + csmaConfig.csmaTries = sTxFrame->mInfo.mTxInfo.mMaxCsmaBackoffs; + csmaConfig.ccaThreshold = sCcaThresholdDbm; - status = RAIL_StartCcaCsmaTx(gRailHandle, sTxFrame->mChannel, txOptions, &csmaConfig, &txSchedulerInfo); + status = RAIL_StartCcaCsmaTx(gRailHandle, sTxFrame->mChannel, txOptions, &csmaConfig, &txSchedulerInfo); + } + else + { + status = RAIL_StartTx(gRailHandle, sTxFrame->mChannel, txOptions, &txSchedulerInfo); + } + + if (status == RAIL_STATUS_NO_ERROR) + { + (void)handlePhyStackEvent(SL_RAIL_UTIL_IEEE802154_STACK_EVENT_TX_STARTED, 0U); + } } else { + // For CSL transmitters (FTDs): + // mTxDelayBaseTime = rx-timestamp (end of sync word) when we received CSL-sync with IEs + // mTxDelay = Delay starting from mTxDelayBaseTime + // + // For CSL receivers (SSEDs): + // mTxDelayBaseTime = timestamp when otPlatRadioTransmit is called + // mTxDelay = Chosen value in the future where transmit is scheduled, so we know exactly + // when to calculate the phase (we can't do this on-the-fly as the packet is going out + // due to platform limitations. see radioScheduleRx) + // + // Note that both use single CCA config, overriding any CCA/CSMA configs from the stack + // #if OPENTHREAD_CONFIG_THREAD_VERSION >= OT_THREAD_VERSION_1_2 - if (sTxFrame->mInfo.mTxInfo.mTxDelay != 0) + RAIL_ScheduleTxConfig_t scheduleTxOptions = {.when = sTxFrame->mInfo.mTxInfo.mTxDelayBaseTime + + sTxFrame->mInfo.mTxInfo.mTxDelay, + .mode = RAIL_TIME_ABSOLUTE, + .txDuringRx = RAIL_SCHEDULED_TX_DURING_RX_POSTPONE_TX}; + + // CSL transmissions don't use CSMA but MAC accounts for single CCA time. + // cslCsmaConfig is set to RAIL_CSMA_CONFIG_SINGLE_CCA above. + status = RAIL_StartScheduledCcaCsmaTx(gRailHandle, + sTxFrame->mChannel, + txOptions, + &scheduleTxOptions, + &cslCsmaConfig, + &txSchedulerInfo); + + if (status == RAIL_STATUS_NO_ERROR) { - // CSL transmissions don't use CSMA but MAC accounts for CCA time. - csmaConfig.ccaThreshold = sCcaThresholdDbm; - - RAIL_ScheduleTxConfig_t scheduleTxOptions = {.when = sTxFrame->mInfo.mTxInfo.mTxDelayBaseTime - + sTxFrame->mInfo.mTxInfo.mTxDelay, - .mode = RAIL_TIME_ABSOLUTE, - .txDuringRx = RAIL_SCHEDULED_TX_DURING_RX_POSTPONE_TX}; - - status = RAIL_StartScheduledCcaCsmaTx(gRailHandle, - sTxFrame->mChannel, - txOptions, - &scheduleTxOptions, - &csmaConfig, - &txSchedulerInfo); - if (status == RAIL_STATUS_NO_ERROR) - { #if RADIO_CONFIG_DEBUG_COUNTERS_SUPPORT - railDebugCounters.mRailEventsScheduledTxTriggeredCount++; + railDebugCounters.mRailEventsScheduledTxTriggeredCount++; #endif - (void)handlePhyStackEvent(SL_RAIL_UTIL_IEEE802154_STACK_EVENT_TX_STARTED, 0U); - } + (void)handlePhyStackEvent(SL_RAIL_UTIL_IEEE802154_STACK_EVENT_TX_STARTED, 0U); } - else #endif - { - status = RAIL_StartTx(gRailHandle, sTxFrame->mChannel, txOptions, &txSchedulerInfo); - if (status == RAIL_STATUS_NO_ERROR) - { - (void)handlePhyStackEvent(SL_RAIL_UTIL_IEEE802154_STACK_EVENT_TX_STARTED, 0U); - } - } } - if (status == RAIL_STATUS_NO_ERROR) { #if RADIO_CONFIG_DEBUG_COUNTERS_SUPPORT @@ -1892,22 +1915,7 @@ otRadioCaps otPlatRadioGetCaps(otInstance *aInstance) { OT_UNUSED_VARIABLE(aInstance); - otRadioCaps capabilities = (OT_RADIO_CAPS_ACK_TIMEOUT | OT_RADIO_CAPS_CSMA_BACKOFF | OT_RADIO_CAPS_ENERGY_SCAN - | OT_RADIO_CAPS_SLEEP_TO_TX); - -#if (OPENTHREAD_CONFIG_THREAD_VERSION >= OT_THREAD_VERSION_1_2) - capabilities |= OT_RADIO_CAPS_TRANSMIT_SEC; -#endif - - // When scheduled tx is required, we support RAIL_StartScheduledCcaCsmaTx - // (delay is indicated in tx frame info set in MAC) - capabilities |= OT_RADIO_CAPS_TRANSMIT_TIMING; - - // When scheduled rx is required, we support RAIL_ScheduleRx in our - // implementation of otPlatRadioReceiveAt - capabilities |= OT_RADIO_CAPS_RECEIVE_TIMING; - - return capabilities; + return sRadioCapabilities; } bool otPlatRadioGetPromiscuous(otInstance *aInstance) @@ -2046,15 +2054,32 @@ void otPlatRadioSetMacKey(otInstance *aInstance, assert(aPrevKey != NULL && aCurrKey != NULL && aNextKey != NULL); - CORE_DECLARE_IRQ_STATE; - CORE_ENTER_ATOMIC(); - sMacKeys[iid].keyId = aKeyId; memcpy(&sMacKeys[iid].keys[MAC_KEY_PREV], aPrevKey, sizeof(otMacKeyMaterial)); memcpy(&sMacKeys[iid].keys[MAC_KEY_CURRENT], aCurrKey, sizeof(otMacKeyMaterial)); memcpy(&sMacKeys[iid].keys[MAC_KEY_NEXT], aNextKey, sizeof(otMacKeyMaterial)); - CORE_EXIT_ATOMIC(); +#if defined(_SILICON_LABS_32B_SERIES_2) && (OPENTHREAD_CONFIG_CRYPTO_LIB == OPENTHREAD_CONFIG_CRYPTO_LIB_PSA) + size_t aKeyLen = 0; + + assert(otPlatCryptoExportKey(sMacKeys[iid].keys[MAC_KEY_PREV].mKeyMaterial.mKeyRef, + sMacKeys[iid].keys[MAC_KEY_PREV].mKeyMaterial.mKey.m8, + sizeof(sMacKeys[iid].keys[MAC_KEY_PREV]), + &aKeyLen) + == OT_ERROR_NONE); + + assert(otPlatCryptoExportKey(sMacKeys[iid].keys[MAC_KEY_CURRENT].mKeyMaterial.mKeyRef, + sMacKeys[iid].keys[MAC_KEY_CURRENT].mKeyMaterial.mKey.m8, + sizeof(sMacKeys[iid].keys[MAC_KEY_CURRENT]), + &aKeyLen) + == OT_ERROR_NONE); + + assert(otPlatCryptoExportKey(sMacKeys[iid].keys[MAC_KEY_NEXT].mKeyMaterial.mKeyRef, + sMacKeys[iid].keys[MAC_KEY_NEXT].mKeyMaterial.mKey.m8, + sizeof(sMacKeys[iid].keys[MAC_KEY_NEXT]), + &aKeyLen) + == OT_ERROR_NONE); +#endif } void otPlatRadioSetMacFrameCounter(otInstance *aInstance, uint32_t aMacFrameCounter) @@ -2229,16 +2254,7 @@ static bool writeIeee802154EnhancedAck(RAIL_Handle_t aRailHandle, receivedFrame.mLength = *initialPktReadBytes - PHY_HEADER_SIZE; enhAckFrame.mPsdu = enhAckPsdu + PHY_HEADER_SIZE; - bool is2015 = otMacFrameIsVersion2015(&receivedFrame); -#ifdef NONCOMPLIANT_ACK_TIMING_WORKAROUND - uint16_t rxToTx = is2015 ? 256 : 192; - if (gTimings.rxToTx != rxToTx) - { - gTimings.rxToTx = rxToTx; - RAIL_SetStateTiming(aRailHandle, &gTimings); - } -#endif - if (!is2015) + if (!otMacFrameIsVersion2015(&receivedFrame)) { return false; } @@ -2306,8 +2322,23 @@ static bool writeIeee802154EnhancedAck(RAIL_Handle_t aRailHandle, #if OPENTHREAD_CONFIG_MAC_CSL_RECEIVER_ENABLE if (sCslPeriod > 0) { + // Calculate time in the future where the SHR is done being sent out + uint32_t ackShrDoneTime = // Currently partially received packet's SHR time + (otPlatAlarmMicroGetNow() + - (packetInfoForEnhAck->packetBytes * OT_RADIO_SYMBOL_TIME * 2) + // PHR of this packet + + (PHY_HEADER_SIZE * OT_RADIO_SYMBOL_TIME * 2) + // Received frame's expected time in the PHR + + (receivedFrame.mLength * OT_RADIO_SYMBOL_TIME * 2) + // rxToTx turnaround time + + sRailIeee802154Config.timings.rxToTx + // PHR time of the ACK + + (PHY_HEADER_SIZE * OT_RADIO_SYMBOL_TIME * 2) + // SHR time of the ACK + + (SHR_SIZE * OT_RADIO_SYMBOL_TIME * 2)); + // Update IE data in the 802.15.4 header with the newest CSL period / phase - otMacFrameSetCslIe(&enhAckFrame, (uint16_t)sCslPeriod, getCslPhase()); + otMacFrameSetCslIe(&enhAckFrame, (uint16_t)sCslPeriod, getCslPhase(ackShrDoneTime)); } #endif @@ -3197,8 +3228,10 @@ static void processNextRxPacket(otInstance *aInstance) #if RADIO_CONFIG_DEBUG_COUNTERS_SUPPORT railDebugCounters.mRailPlatRadioReceiveDoneCbCount++; #endif - sReceivePacket[index].state = BUFFER_IS_FREE; } + + sReceivePacket[index].state = BUFFER_IS_FREE; + otSysEventSignalPending(); } #if SL_OPENTHREAD_RADIO_RX_BUFFER_COUNT > 1 @@ -3254,6 +3287,9 @@ static void processTxComplete(otInstance *aInstance) else #endif { + // Clear any internally-set txDelays so future transmits are not affected. + sTxFrame->mInfo.mTxInfo.mTxDelayBaseTime = 0; + sTxFrame->mInfo.mTxInfo.mTxDelay = 0; otPlatRadioTxDone(aInstance, sTxFrame, ackFrame, txStatus); } diff --git a/src/src/radio_extension.c b/src/src/radio_extension.c index dfa82200..0b503518 100644 --- a/src/src/radio_extension.c +++ b/src/src/radio_extension.c @@ -501,7 +501,7 @@ otError otPlatRadioExtensionGetRadioCounters(efr32RadioCounters *aRadioCounters) VerifyOrExit(aRadioCounters != NULL, error = OT_ERROR_INVALID_ARGS); *aRadioCounters = railDebugCounters; #else - error = OT_ERROR_NOT_IMPLEMENTED; + ExitNow(error = OT_ERROR_NOT_IMPLEMENTED); #endif exit: diff --git a/src/src/system.c b/src/src/system.c index 096a3906..e7e4ec1c 100644 --- a/src/src/system.c +++ b/src/src/system.c @@ -47,6 +47,13 @@ #include "rail.h" #include "sl_mpu.h" #include "sl_sleeptimer.h" +#if OPENTHREAD_CONFIG_HEAP_EXTERNAL_ENABLE +#include "sl_malloc.h" +#include "openthread/heap.h" +#endif + +#include "platform-efr32.h" +#include "sl_openthread.h" #if defined(SL_COMPONENT_CATALOG_PRESENT) #include "sl_component_catalog.h" @@ -68,14 +75,6 @@ #include "sl_rcp_gp_interface.h" #endif // SL_CATALOG_OT_RCP_GP_INTERFACE_PRESENT -#if OPENTHREAD_CONFIG_HEAP_EXTERNAL_ENABLE -#include "sl_malloc.h" -#include "openthread/heap.h" -#endif - -#include "platform-efr32.h" -#include "sl_openthread.h" - #define USE_EFR32_LOG (OPENTHREAD_CONFIG_LOG_OUTPUT == OPENTHREAD_CONFIG_LOG_OUTPUT_PLATFORM_DEFINED) otInstance *sInstance; diff --git a/third_party/silabs/gecko_sdk b/third_party/silabs/gecko_sdk index 2ba59bf9..1228a952 160000 --- a/third_party/silabs/gecko_sdk +++ b/third_party/silabs/gecko_sdk @@ -1 +1 @@ -Subproject commit 2ba59bf969e94a0544e73e501d876722790f7a89 +Subproject commit 1228a95262ee099a21c6be4d35224479c8e4dde2