/***************************************************************************//** * @file * @brief Chip Errata Workarounds ******************************************************************************* * # License * Copyright 2018 Silicon Laboratories Inc. www.silabs.com ******************************************************************************* * * SPDX-License-Identifier: Zlib * * The licensor of this software is Silicon Laboratories Inc. * * This software is provided 'as-is', without any express or implied * warranty. In no event will the authors be held liable for any damages * arising from the use of this software. * * Permission is granted to anyone to use this software for any purpose, * including commercial applications, and to alter it and redistribute it * freely, subject to the following restrictions: * * 1. The origin of this software must not be misrepresented; you must not * claim that you wrote the original software. If you use this software * in a product, an acknowledgment in the product documentation would be * appreciated but is not required. * 2. Altered source versions must be plainly marked as such, and must not be * misrepresented as being the original software. * 3. This notice may not be removed or altered from any source distribution. * ******************************************************************************/ #ifndef EM_CHIP_H #define EM_CHIP_H #include "em_device.h" #include "sl_common.h" #if defined(_SILICON_LABS_32B_SERIES) && (_SILICON_LABS_32B_SERIES <= 2) #include "em_system.h" #endif #include "em_bus.h" #if defined(_SILICON_LABS_GECKO_INTERNAL_SDID_80) #include "em_gpio.h" #endif #ifdef __cplusplus extern "C" { #endif /***************************************************************************//** * @addtogroup chip CHIP - Chip Errata Workarounds * @brief Chip errata workaround APIs * @details * API to apply chip errata workarounds at initialization and reset. * @{ ******************************************************************************/ /**************************************************************************//** * @brief * Chip initialization routine for revision errata workarounds. * * @note * This function must be called immediately in main(). * * This initialization function configures the device to a state * as similar to later revisions as possible to improve software compatibility * with newer parts. See the device-specific errata for details. *****************************************************************************/ __STATIC_INLINE void CHIP_Init(void) { #if defined(MSC_CACHECMD_INVCACHE) MSC->CACHECMD = MSC_CACHECMD_INVCACHE; #elif defined(MSC_CMD_INVCACHE) MSC->CMD = MSC_CMD_INVCACHE; #endif #if defined(_SILICON_LABS_32B_SERIES_0) && defined(_EFM32_GECKO_FAMILY) uint32_t rev; SYSTEM_ChipRevision_TypeDef chipRev; volatile uint32_t *reg; rev = *(volatile uint32_t *)(0x0FE081FC); /* Engineering Sample calibration setup. */ if ((rev >> 24) == 0) { reg = (volatile uint32_t *)0x400CA00C; *reg &= ~(0x70UL); /* DREG */ reg = (volatile uint32_t *)0x400C6020; *reg &= ~(0xE0000000UL); *reg |= ~(7UL << 25); } if ((rev >> 24) <= 3) { /* DREG */ reg = (volatile uint32_t *)0x400C6020; *reg &= ~(0x00001F80UL); /* Update CMU reset values. */ reg = (volatile uint32_t *)0x400C8040; *reg = 0; reg = (volatile uint32_t *)0x400C8044; *reg = 0; reg = (volatile uint32_t *)0x400C8058; *reg = 0; reg = (volatile uint32_t *)0x400C8060; *reg = 0; reg = (volatile uint32_t *)0x400C8078; *reg = 0; } SYSTEM_ChipRevisionGet(&chipRev); if (chipRev.major == 0x01) { /* Rev A errata handling for EM2/3. Must enable DMA clock to get EM2/3 */ /* to work. This will be fixed in later chip revisions and is only needed for rev A. */ if (chipRev.minor == 00) { reg = (volatile uint32_t *)0x400C8040; *reg |= 0x2; } /* Rev A+B errata handling for I2C when using EM2/3. USART0 clock must be enabled */ /* after waking up from EM2/EM3 to get I2C to work. This will be fixed in */ /* later chip revisions and is only needed for rev A+B. */ if (chipRev.minor <= 0x01) { reg = (volatile uint32_t *)0x400C8044; *reg |= 0x1; } } /* Ensure correct ADC/DAC calibration value. */ rev = *(volatile uint32_t *)0x0FE081F0; if (rev < 0x4C8ABA00) { uint32_t cal; /* Enable ADC/DAC clocks. */ reg = (volatile uint32_t *)0x400C8044UL; *reg |= (1 << 14 | 1 << 11); /* Retrive calibration values. */ cal = ((*(volatile uint32_t *)(0x0FE081B4UL) & 0x00007F00UL) >> 8) << 24; cal |= ((*(volatile uint32_t *)(0x0FE081B4UL) & 0x0000007FUL) >> 0) << 16; cal |= ((*(volatile uint32_t *)(0x0FE081B4UL) & 0x00007F00UL) >> 8) << 8; cal |= ((*(volatile uint32_t *)(0x0FE081B4UL) & 0x0000007FUL) >> 0) << 0; /* ADC0->CAL = 1.25 reference. */ reg = (volatile uint32_t *)0x40002034UL; *reg = cal; /* DAC0->CAL = 1.25 reference. */ reg = (volatile uint32_t *)(0x4000402CUL); cal = *(volatile uint32_t *)0x0FE081C8UL; *reg = cal; /* Turn off ADC/DAC clocks. */ reg = (volatile uint32_t *)0x400C8044UL; *reg &= ~(1 << 14 | 1 << 11); } #endif #if defined(_SILICON_LABS_32B_SERIES_0) && defined(_EFM32_GIANT_FAMILY) /****************************/ /* Fix for errata CMU_E113. */ uint8_t prodRev; SYSTEM_ChipRevision_TypeDef chipRev; prodRev = SYSTEM_GetProdRev(); SYSTEM_ChipRevisionGet(&chipRev); // All Giant and Leopard parts except Leopard Rev E if ((prodRev >= 16) && (chipRev.minor >= 3) && !((chipRev.major == 2) && (chipRev.minor == 4))) { /* This fixes an issue with the LFXO on high temperatures. */ *(volatile uint32_t*)0x400C80C0 = (*(volatile uint32_t*)0x400C80C0 & ~(1 << 6) ) | (1 << 4); } #endif #if defined(_SILICON_LABS_32B_SERIES_0) && defined(_EFM32_HAPPY_FAMILY) uint8_t prodRev; prodRev = SYSTEM_GetProdRev(); if (prodRev <= 129) { /* This fixes a mistaken internal connection between PC0 and PC4. */ /* This disables an internal pull-down on PC4. */ *(volatile uint32_t*)(0x400C6018) = (1 << 26) | (5 << 0); /* This disables an internal LDO test signal driving PC4. */ *(volatile uint32_t*)(0x400C80E4) &= ~(1 << 24); } #endif #if defined(_SILICON_LABS_GECKO_INTERNAL_SDID_80) /**************************** * Fixes for errata GPIO_E201 (slewrate) and * HFXO high-temperature oscillator startup robustness fix. */ uint32_t port; uint32_t clkEn; uint8_t prodRev; const uint32_t setVal = (0x5 << _GPIO_P_CTRL_SLEWRATEALT_SHIFT) | (0x5 << _GPIO_P_CTRL_SLEWRATE_SHIFT); const uint32_t resetVal = _GPIO_P_CTRL_RESETVALUE & ~(_GPIO_P_CTRL_SLEWRATE_MASK | _GPIO_P_CTRL_SLEWRATEALT_MASK); prodRev = SYSTEM_GetProdRev(); SYSTEM_ChipRevision_TypeDef chipRev; SYSTEM_ChipRevisionGet(&chipRev); /* This errata is fixed in hardware from PRODREV 0x8F. */ if (prodRev < 0x8F) { /* Fixes for errata GPIO_E201 (slewrate). */ /* Save HFBUSCLK enable state and enable GPIO clock. */ clkEn = CMU->HFBUSCLKEN0; CMU->HFBUSCLKEN0 = clkEn | CMU_HFBUSCLKEN0_GPIO; /* Update slewrate. */ for (port = 0; port <= GPIO_PORT_MAX; port++) { GPIO->P[port].CTRL = setVal | resetVal; } /* Restore HFBUSCLK enable state. */ CMU->HFBUSCLKEN0 = clkEn; } /* This errata is fixed in hardware from PRODREV 0x90. */ if (prodRev < 0x90) { /* HFXO high-temperature oscillator startup robustness fix. */ CMU->HFXOSTARTUPCTRL = (CMU->HFXOSTARTUPCTRL & ~_CMU_HFXOSTARTUPCTRL_IBTRIMXOCORE_MASK) | (0x20 << _CMU_HFXOSTARTUPCTRL_IBTRIMXOCORE_SHIFT); } if (chipRev.major == 0x01) { /* Fix for errata EMU_E210 - Potential Power-Down When Entering EM2 */ *(volatile uint32_t *)(EMU_BASE + 0x164) |= 0x4; } /**************************** * Fix for errata DCDC_E206. * Disable bypass limit enabled temporarily in SystemInit() errata * workaround. */ BUS_RegBitWrite(&EMU->DCDCCLIMCTRL, _EMU_DCDCCLIMCTRL_BYPLIMEN_SHIFT, 0); #endif #if defined(_SILICON_LABS_GECKO_INTERNAL_SDID_84) uint8_t prodRev = SYSTEM_GetProdRev(); /* EM2 current fixes for early samples. */ if (prodRev == 0U) { *(volatile uint32_t *)(EMU_BASE + 0x190UL) = 0x0000ADE8UL; *(volatile uint32_t *)(EMU_BASE + 0x198UL) |= (0x1UL << 2); *(volatile uint32_t *)(EMU_BASE + 0x190UL) = 0x0; } if (prodRev < 2U) { *(volatile uint32_t *)(EMU_BASE + 0x164UL) |= (0x1UL << 13); } /* Set optimal LFRCOCTRL VREFUPDATE and enable duty cycling of VREF. */ CMU->LFRCOCTRL = (CMU->LFRCOCTRL & ~_CMU_LFRCOCTRL_VREFUPDATE_MASK) | CMU_LFRCOCTRL_VREFUPDATE_64CYCLES | CMU_LFRCOCTRL_ENVREF; #endif #if defined(_SILICON_LABS_32B_SERIES_1) \ && defined(_EFR_DEVICE) && (_SILICON_LABS_GECKO_INTERNAL_SDID >= 84) MSC->CTRL |= 0x1UL << 8; #endif #if defined(_SILICON_LABS_GECKO_INTERNAL_SDID_89) SYSTEM_ChipRevision_TypeDef chipRev; SYSTEM_ChipRevisionGet(&chipRev); if ((chipRev.major > 1) || (chipRev.minor >= 3)) { /* PLFRCO trim values */ *(volatile uint32_t *)(CMU_BASE + 0x28CUL) = 608; *(volatile uint32_t *)(CMU_BASE + 0x290UL) = 356250; *(volatile uint32_t *)(CMU_BASE + 0x2F0UL) = 0x04000118; *(volatile uint32_t *)(CMU_BASE + 0x2F8UL) = 0x08328400; } #endif /* Charge redist setup (fixed value): LCD->DBGCTRL.CHGRDSTSTR = 1 (reset: 0). */ #if defined(_LCD_DISPCTRL_CHGRDST_MASK) #if defined(_SILICON_LABS_32B_SERIES_1) CMU->HFBUSCLKEN0 |= CMU_HFBUSCLKEN0_LE; CMU->LFACLKEN0 |= CMU_LFACLKEN0_LCD; *(volatile uint32_t *)(LCD_BASE + 0x034) |= (0x1UL << 12); CMU->LFACLKEN0 &= ~CMU_LFACLKEN0_LCD; CMU->HFBUSCLKEN0 &= ~CMU_HFBUSCLKEN0_LE; #endif #endif #if defined(_SILICON_LABS_32B_SERIES_1) \ && !defined(_SILICON_LABS_GECKO_INTERNAL_SDID_80) \ && !defined(ERRATA_FIX_EMU_E220_DECBOD_IGNORE) /* First part of the EMU_E220 DECBOD Errata fix. DECBOD Reset can occur * during voltage scaling after EM2/3 wakeup. Second part is in em_emu.c */ *(volatile uint32_t *)(EMU_BASE + 0x1A4) |= 0x1f << 10; #endif #if defined(_SILICON_LABS_32B_SERIES_2_CONFIG_1) SYSTEM_ChipRevision_TypeDef chipRev; SYSTEM_ChipRevisionGet(&chipRev); if (chipRev.major == 0x01 && (HFXO0->STATUS & HFXO_STATUS_ENS) == 0U) { /* Change HFXO default peak detector settings. */ *(volatile uint32_t*)(HFXO0_BASE + 0x34U) = (*(volatile uint32_t*)(HFXO0_BASE + 0x34U) & 0xFF8000FFU) | 0x00178500U; /* Change HFXO low power control settings. */ *(volatile uint32_t*)(HFXO0_BASE + 0x30U) = (*(volatile uint32_t*)(HFXO0_BASE + 0x30U) & 0xFFFF0FFFU) | 0x0000C000U; /* Change default SQBUF bias current. */ *(volatile uint32_t*)(HFXO0_BASE + 0x30U) |= 0x700; } if (chipRev.major == 0x01 && chipRev.minor == 0x0) { /* Trigger RAM read for each RAM instance */ volatile uint32_t *dmem = (volatile uint32_t *) DMEM_RAM0_RAM_MEM_BASE; for (uint32_t i = 0U; i < DMEM_NUM_BANK; i++) { // Force memory read *dmem; dmem += (DMEM_BANK0_SIZE / 4U); } } /* Set TRACE clock to intended reset value. */ CMU->TRACECLKCTRL = (CMU->TRACECLKCTRL & ~_CMU_TRACECLKCTRL_CLKSEL_MASK) | CMU_TRACECLKCTRL_CLKSEL_HFRCOEM23; #endif #if defined(_SILICON_LABS_GECKO_INTERNAL_SDID_205) #if defined(SL_TRUSTZONE_SECURE) #define HFRCO_CLK_CFG_CLR_ADDR (0x40012020UL) #else #define HFRCO_CLK_CFG_CLR_ADDR (0x50012020UL) #endif #define HFRCO_CLK_CFG_CLKOUTDIS0 (0x4UL) if (SYSTEM_GetProdRev() == 1) { bool hfrcoClkIsOff = (CMU->CLKEN0 & CMU_CLKEN0_HFRCO0) == 0; CMU->CLKEN0_SET = CMU_CLKEN0_HFRCO0; /* Enable HFRCO CLKOUT0. */ *(volatile uint32_t*)(HFRCO_CLK_CFG_CLR_ADDR) = HFRCO_CLK_CFG_CLKOUTDIS0; if (hfrcoClkIsOff) { CMU->CLKEN0_CLR = CMU_CLKEN0_HFRCO0; } } #endif /* PM-3503 */ #if defined(_SILICON_LABS_GECKO_INTERNAL_SDID_210) { bool syscfgClkIsOff = ((CMU->CLKEN0 & CMU_CLKEN0_SYSCFG) == 0); CMU->CLKEN0_SET = CMU_CLKEN0_SYSCFG; bool dcdcClkIsOff = ((CMU->CLKEN0 & CMU_CLKEN0_DCDC) == 0); CMU->CLKEN0_SET = CMU_CLKEN0_DCDC; bool dcdcIsLock = ((DCDC->LOCKSTATUS & DCDC_LOCKSTATUS_LOCK_LOCKED) != 0); DCDC->LOCK = DCDC_LOCK_LOCKKEY_UNLOCKKEY; while (DCDC->SYNCBUSY & DCDC_SYNCBUSY_CTRL) { /* Wait for previous synchronization to finish */ } DCDC->CTRL_CLR = DCDC_CTRL_MODE; while ((DCDC->STATUS & DCDC_STATUS_BYPSW) == 0U) { /* Wait for BYPASS switch enable. */ } if (dcdcIsLock) { DCDC->LOCK = ~DCDC_LOCK_LOCKKEY_UNLOCKKEY; } if (dcdcClkIsOff) { CMU->CLKEN0_CLR = CMU_CLKEN0_DCDC; } if (syscfgClkIsOff) { CMU->CLKEN0_CLR = CMU_CLKEN0_SYSCFG; } } #endif /* PM-5163 */ #if defined(_SILICON_LABS_GECKO_INTERNAL_SDID_215) \ && defined(_SILICON_LABS_EFR32_2G4HZ_HP_PA_PRESENT) \ && (_SILICON_LABS_EFR32_2G4HZ_HP_PA_MAX_OUTPUT_DBM == 20) SYSTEM_ChipRevision_TypeDef chipRev; SYSTEM_ChipRevisionGet(&chipRev); if (chipRev.major == 0x01 && chipRev.minor == 0x00) { bool hfxo0ClkIsOff = (CMU->CLKEN0 & CMU_CLKEN0_HFXO0) == 0; CMU->CLKEN0_SET = CMU_CLKEN0_HFXO0; *(volatile uint32_t*)(HFXO0_BASE + 0x0034UL) = (*(volatile uint32_t*)(HFXO0_BASE + 0x0034UL) & 0xE3FFFFFFUL) | 0x0C000000UL; if (hfxo0ClkIsOff) { CMU->CLKEN0_CLR = CMU_CLKEN0_HFXO0; } } #endif } /**************************************************************************//** * @brief * Chip reset routine with errata workarounds. * * @note * This function should be called to reset the chip. It does not return. * * This function applies any errata workarounds needed to cleanly reset the * device and then performs a system reset. See the device-specific errata for * details. *****************************************************************************/ __STATIC_INLINE void CHIP_Reset(void) { #if defined(_EFR_DEVICE) && defined(_SILICON_LABS_GECKO_INTERNAL_SDID_80) /**************************** * Workaround for errata DCDC_E206. * Disable radio interference minimization features when resetting */ // Ensure access to EMU registers EMU->LOCK = EMU_LOCK_LOCKKEY_UNLOCK; EMU->PWRLOCK = EMU_PWRLOCK_LOCKKEY_LOCK; // No need to do anything if the DCDC is not powering DVDD if ((EMU->PWRCFG & _EMU_PWRCFG_PWRCFG_MASK) == EMU_PWRCFG_PWRCFG_DCDCTODVDD) { // Make sure radio cannot accidentally re-enable features *(volatile uint32_t *)(0x40084040UL) = 0x1UL; // If DCDC is in use, disable features uint32_t dcdcMode = EMU->DCDCCTRL & _EMU_DCDCCTRL_DCDCMODE_MASK; if ((dcdcMode == EMU_DCDCCTRL_DCDCMODE_LOWNOISE) || (dcdcMode == EMU_DCDCCTRL_DCDCMODE_LOWPOWER)) { BUS_RegBitWrite((volatile uint32_t *)(0x400E3060UL), 28UL, 0); BUS_RegBitWrite((volatile uint32_t *)(0x400E3074UL), 0, 0); } } #endif NVIC_SystemReset(); } /** @} (end addtogroup chip) */ #ifdef __cplusplus } #endif #endif /* EM_CHIP_H */