/* * Copyright (C) 2017 Freie Universität Berlin * 2017 OTA keys S.A. * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @ingroup cpu_stm32 * @{ * * @file * @brief Implementation of common STM32 clock configuration functions * * @author Hauke Petersen * @author Vincent Dupont * @} */ #include "cpu.h" #include "stmclk.h" #include "periph_conf.h" #if defined(CPU_FAM_STM32L4) || defined(CPU_FAM_STM32F7) || \ defined(CPU_FAM_STM32WB) || defined(CPU_FAM_STM32G4) || \ defined(CPU_FAM_STM32G0) || defined(CPU_FAM_STM32L5) || \ defined(CPU_FAM_STM32WL) #define REG_PWR_CR CR1 #define BIT_CR_DBP PWR_CR1_DBP #else #define REG_PWR_CR CR #define BIT_CR_DBP PWR_CR_DBP #endif #if defined (CPU_FAM_STM32L0) || defined(CPU_FAM_STM32L1) #define REG_LSE CSR #define BIT_LSEON RCC_CSR_LSEON #define BIT_LSERDY RCC_CSR_LSERDY #else #define REG_LSE BDCR #define BIT_LSEON RCC_BDCR_LSEON #define BIT_LSERDY RCC_BDCR_LSERDY #endif #if defined (CPU_FAM_STM32WB) #define RCC_CFGR_SWS_HSI RCC_CFGR_SWS_0 #define RCC_CSR_LSION RCC_CSR_LSI1ON #define RCC_CSR_LSIRDY RCC_CSR_LSI1RDY #endif #if defined(CPU_FAM_STM32L5) #define RCC_CFGR_SWS_HSI RCC_CFGR_SWS_0 #endif void stmclk_enable_hsi(void) { RCC->CR |= RCC_CR_HSION; while (!(RCC->CR & RCC_CR_HSIRDY)) {} } void stmclk_disable_hsi(void) { RCC->CR &= ~RCC_CR_HSION; } void stmclk_enable_lfclk(void) { if (IS_ACTIVE(CONFIG_BOARD_HAS_LSE)) { stmclk_dbp_unlock(); RCC->REG_LSE |= BIT_LSEON; while (!(RCC->REG_LSE & BIT_LSERDY)) {} /* Set LSE system clock enable bit. This is required if LSE is to be used by USARTx, LPUARTx, LPTIMx, TIMx, RNG, system LSCO, MCO, MSI PLL mode */ #if defined(CPU_FAM_STM32WL) RCC->BDCR |= RCC_BDCR_LSESYSEN; while (!(RCC->BDCR & RCC_BDCR_LSESYSRDY)) {} #endif stmclk_dbp_lock(); } else { RCC->CSR |= RCC_CSR_LSION; while (!(RCC->CSR & RCC_CSR_LSIRDY)) {} } } void stmclk_disable_lfclk(void) { if (IS_ACTIVE(CONFIG_BOARD_HAS_LSE)) { stmclk_dbp_unlock(); RCC->REG_LSE &= ~(BIT_LSEON); while (!(RCC->REG_LSE & BIT_LSERDY)) {} stmclk_dbp_lock(); } else { RCC->CSR &= ~(RCC_CSR_LSION); } } void stmclk_dbp_unlock(void) { PWR->REG_PWR_CR |= BIT_CR_DBP; } void stmclk_dbp_lock(void) { PWR->REG_PWR_CR &= ~(BIT_CR_DBP); }