/* * Copyright (C) 2013 INRIA * 2014 Freie Universität Berlin * 2016 TriaGnoSys GmbH * 2018 Kaspar Schleiser * 2018 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 the kernel cpu functions * * @author Stefan Pfeiffer * @author Alaeddine Weslati * @author Thomas Eichinger * @author Hauke Petersen * @author Nick van IJzendoorn * @author Víctor Ariño * @author Kaspar Schleiser * @author Vincent Dupont * @author Oleg Artamonov * @author Francisco Molina * * @} */ #include "cpu.h" #include "stdio_base.h" #include "stmclk.h" #include "periph_cpu.h" #include "periph/init.h" #include "board.h" #if defined (CPU_FAM_STM32L4) #define BIT_APB_PWREN RCC_APB1ENR1_PWREN #else #define BIT_APB_PWREN RCC_APB1ENR_PWREN #endif #if defined(CPU_FAM_STM32F0) || defined(CPU_FAM_STM32F1) || \ defined(CPU_FAM_STM32F2) || defined(CPU_FAM_STM32F3) || \ defined(CPU_FAM_STM32F4) || defined(CPU_FAM_STM32F7) || \ defined(CPU_FAM_STM32L1) #define STM32_CPU_MAX_GPIOS (12U) #if defined(CPU_FAM_STM32L1) #define GPIO_CLK (AHB) #define GPIO_CLK_ENR (RCC->AHBENR) #define GPIO_CLK_ENR_MASK (0x0000FFFF) #elif defined(CPU_FAM_STM32F0) || defined(CPU_FAM_STM32F3) #define GPIO_CLK (AHB) #define GPIO_CLK_ENR (RCC->AHBENR) #define GPIO_CLK_ENR_MASK (0xFFFF0000) #elif defined(CPU_FAM_STM32F2) || defined(CPU_FAM_STM32F4) || \ defined(CPU_FAM_STM32F7) #define GPIO_CLK (AHB1) #define GPIO_CLK_ENR (RCC->AHB1ENR) #define GPIO_CLK_ENR_MASK (0x0000FFFF) #elif defined(CPU_FAM_STM32F1) #define GPIO_CLK (APB2) #define GPIO_CLK_ENR (RCC->APB2ENR) #define GPIO_CLK_ENR_MASK (0x000001FC) #endif #ifndef DISABLE_JTAG #define DISABLE_JTAG 0 #endif /** * @brief Initialize gpio to AIN * * stm32f need to have all there pins initialized to AIN so the consumption * of the input Schmitt trigger is saved when running in STOP mode. * * @see https://comm.eefocus.com/media/download/index/id-1013834 */ static void _gpio_init_ain(void) { uint32_t ahb_gpio_clocks; /* enable GPIO clock and save GPIO clock configuration */ ahb_gpio_clocks = GPIO_CLK_ENR & GPIO_CLK_ENR_MASK; periph_clk_en(GPIO_CLK, GPIO_CLK_ENR_MASK); /* switch all GPIOs to AIN mode to minimize power consumption */ for (uint8_t i = 0; i < STM32_CPU_MAX_GPIOS; i++) { GPIO_TypeDef *port; port = (GPIO_TypeDef *)(GPIOA_BASE + i*(GPIOB_BASE - GPIOA_BASE)); if (IS_GPIO_ALL_INSTANCE(port)) { if (!DISABLE_JTAG) { #if defined(CPU_FAM_STM32F1) switch (i) { /* preserve JTAG pins on PORTA and PORTB */ case 0: port->CRL = GPIO_CRL_CNF; port->CRH = GPIO_CRH_CNF & 0x000FFFFF; break; case 1: port->CRL = GPIO_CRL_CNF & 0xFFF00FFF; port->CRH = GPIO_CRH_CNF; break; default: port->CRL = GPIO_CRL_CNF; port->CRH = GPIO_CRH_CNF; break; } #else /* ! defined(CPU_FAM_STM32F1) */ switch (i) { /* preserve JTAG pins on PORTA and PORTB */ case 0: port->MODER = 0xABFFFFFF; break; case 1: port->MODER = 0xFFFFFEBF; break; default: port->MODER = 0xFFFFFFFF; break; } #endif /* defined(CPU_FAM_STM32F1) */ } else { #if defined(CPU_FAM_STM32F1) port->CRL = GPIO_CRL_CNF; port->CRH = GPIO_CRH_CNF; #else port->MODER = 0xFFFFFFFF; #endif } } } /* restore GPIO clocks */ periph_clk_en(GPIO_CLK, ahb_gpio_clocks); } #endif void cpu_init(void) { /* initialize the Cortex-M core */ cortexm_init(); /* enable PWR module */ #ifndef CPU_FAM_STM32WB periph_clk_en(APB1, BIT_APB_PWREN); #endif /* initialize the system clock as configured in the periph_conf.h */ stmclk_init_sysclk(); #if defined(CPU_FAM_STM32F0) || defined(CPU_FAM_STM32F1) || \ defined(CPU_FAM_STM32F2) || defined(CPU_FAM_STM32F3) || \ defined(CPU_FAM_STM32F4) || defined(CPU_FAM_STM32F7) || \ defined(CPU_FAM_STM32L1) _gpio_init_ain(); #endif #ifdef MODULE_PERIPH_DMA /* initialize DMA streams */ dma_init(); #endif /* initialize stdio prior to periph_init() to allow use of DEBUG() there */ stdio_init(); #ifdef STM32F1_DISABLE_JTAG RCC->APB2ENR |= RCC_APB2ENR_AFIOEN; AFIO->MAPR |= AFIO_MAPR_SWJ_CFG_JTAGDISABLE; #endif /* trigger static peripheral initialization */ periph_init(); }