2020-05-03 14:35:01 +02:00
|
|
|
|
/*
|
|
|
|
|
* Copyright (C) 2013 INRIA
|
|
|
|
|
* 2014 Freie Universität Berlin
|
|
|
|
|
* 2016 TriaGnoSys GmbH
|
|
|
|
|
* 2018 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
|
* 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.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/**
|
2020-05-03 17:17:54 +02:00
|
|
|
|
* @ingroup cpu_stm32
|
2020-05-03 14:35:01 +02:00
|
|
|
|
* @{
|
|
|
|
|
*
|
|
|
|
|
* @file
|
|
|
|
|
* @brief Implementation of the kernel cpu functions
|
|
|
|
|
*
|
|
|
|
|
* @author Stefan Pfeiffer <stefan.pfeiffer@fu-berlin.de>
|
|
|
|
|
* @author Alaeddine Weslati <alaeddine.weslati@inria.fr>
|
|
|
|
|
* @author Thomas Eichinger <thomas.eichinger@fu-berlin.de>
|
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
|
* @author Nick van IJzendoorn <nijzendoorn@engineering-spirit.nl>
|
|
|
|
|
* @author Víctor Ariño <victor.arino@zii.aero>
|
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
|
* @author Vincent Dupont <vincent@otakeys.com>
|
|
|
|
|
* @author Oleg Artamonov <oleg@unwds.com>
|
|
|
|
|
* @author Francisco Molina <francisco.molina@inria.cl>
|
|
|
|
|
*
|
|
|
|
|
* @}
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "cpu.h"
|
2023-01-02 18:08:35 +01:00
|
|
|
|
#include "kernel_init.h"
|
2020-05-03 14:35:01 +02:00
|
|
|
|
#include "stdio_base.h"
|
|
|
|
|
#include "stmclk.h"
|
|
|
|
|
#include "periph_cpu.h"
|
|
|
|
|
#include "periph/init.h"
|
2021-07-07 13:41:56 +02:00
|
|
|
|
#include "periph/gpio.h"
|
2021-10-07 21:08:27 +02:00
|
|
|
|
#include "periph/vbat.h"
|
2020-05-03 14:35:01 +02:00
|
|
|
|
#include "board.h"
|
2021-10-12 20:00:19 +02:00
|
|
|
|
#include "pm_layered.h"
|
2020-05-03 14:35:01 +02:00
|
|
|
|
|
2020-09-28 15:57:01 +02:00
|
|
|
|
#if defined (CPU_FAM_STM32L4) || defined (CPU_FAM_STM32G4) || \
|
|
|
|
|
defined(CPU_FAM_STM32L5)
|
2020-05-03 14:35:01 +02:00
|
|
|
|
#define BIT_APB_PWREN RCC_APB1ENR1_PWREN
|
2024-01-25 22:45:17 +01:00
|
|
|
|
#elif defined(CPU_FAM_STM32G0) || defined(CPU_FAM_STM32C0)
|
2020-05-03 22:22:10 +02:00
|
|
|
|
#define BIT_APB_PWREN RCC_APBENR1_PWREN
|
2020-08-03 00:19:58 +02:00
|
|
|
|
#elif !defined(CPU_FAM_STM32MP1)
|
2020-05-03 14:35:01 +02:00
|
|
|
|
#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)
|
2021-03-05 01:00:00 +01:00
|
|
|
|
#elif defined(CPU_FAM_STM32WL)
|
|
|
|
|
#define GPIO_CLK (AHB2)
|
|
|
|
|
#define GPIO_CLK_ENR (RCC->AHB2ENR)
|
|
|
|
|
#define GPIO_CLK_ENR_MASK (0x00000087)
|
2020-05-03 14:35:01 +02:00
|
|
|
|
#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:
|
2020-05-26 16:26:49 +02:00
|
|
|
|
port->CRL = GPIO_CRL_CNF;
|
|
|
|
|
port->CRH = GPIO_CRH_CNF & 0x000FFFFF;
|
2020-05-03 14:35:01 +02:00
|
|
|
|
break;
|
|
|
|
|
case 1:
|
2020-05-26 16:26:49 +02:00
|
|
|
|
port->CRL = GPIO_CRL_CNF & 0xFFF00FFF;
|
|
|
|
|
port->CRH = GPIO_CRH_CNF;
|
2020-05-03 14:35:01 +02:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2020-05-26 16:26:49 +02:00
|
|
|
|
port->CRL = GPIO_CRL_CNF;
|
|
|
|
|
port->CRH = GPIO_CRH_CNF;
|
2020-05-03 14:35:01 +02:00
|
|
|
|
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)
|
2020-05-26 16:26:49 +02:00
|
|
|
|
port->CRL = GPIO_CRL_CNF;
|
|
|
|
|
port->CRH = GPIO_CRH_CNF;
|
2020-05-03 14:35:01 +02:00
|
|
|
|
#else
|
|
|
|
|
port->MODER = 0xFFFFFFFF;
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* restore GPIO clocks */
|
|
|
|
|
periph_clk_en(GPIO_CLK, ahb_gpio_clocks);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-10-12 20:00:19 +02:00
|
|
|
|
/**
|
|
|
|
|
* @brief get the value of a register in a glitch resistant fashion
|
|
|
|
|
*
|
|
|
|
|
* This very teniously avoids optimization, even optimized it's better than
|
|
|
|
|
* nothing but periodic review should establish that it doesn't get optimized.
|
|
|
|
|
*/
|
2022-11-10 14:24:11 +01:00
|
|
|
|
MAYBE_UNUSED
|
2021-10-12 20:00:19 +02:00
|
|
|
|
__attribute__((always_inline))
|
|
|
|
|
static inline uint32_t _multi_read_reg32(volatile uint32_t *addr, bool *glitch)
|
|
|
|
|
{
|
|
|
|
|
uint32_t value = *addr;
|
|
|
|
|
// cppcheck-suppress duplicateExpression
|
|
|
|
|
// cppcheck-suppress knownConditionTrueFalse
|
|
|
|
|
if (*addr != value || *addr != value) {
|
|
|
|
|
/* (reason: volatile pointer forces multiple reads for glitch resistance,
|
|
|
|
|
glitch may force different value) */
|
|
|
|
|
*glitch = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief Check RDP level is what the designer intended.
|
|
|
|
|
*
|
|
|
|
|
* RDP stands for "ReaDout Protection."
|
|
|
|
|
*
|
|
|
|
|
* The STM32L4 readout protection feature offers three levels of protection
|
|
|
|
|
* for all SRAM2 and Flash memory as well as the backup registers:
|
|
|
|
|
*
|
|
|
|
|
* - Level 0 (RDP0) means “no protection”. This is the factory default. Read,
|
|
|
|
|
* Write and Erase operations are permitted in the SRAM2 and Flash memory
|
|
|
|
|
* as well as the backup registers. Option bytes are changeable in Level 0.
|
|
|
|
|
*
|
|
|
|
|
* - Level 1 (RDP1) ensures read protection of the chip’s memories which
|
|
|
|
|
* includes the Flash memory and the backup registers as well as the SRAM2
|
|
|
|
|
* content. Whenever a debugger access is detected or Boot mode is not set
|
|
|
|
|
* to a Flash memory area, any access to the Flash memory, the backup
|
|
|
|
|
* registers or to the SRAM2 generates a system hard fault which blocks all
|
|
|
|
|
* code execution until the next power-on reset. Option bytes can still be
|
|
|
|
|
* modified in Level 1.
|
|
|
|
|
*
|
|
|
|
|
* - Level 2 (RDP2) provides the same protection features for the SRAM2,
|
|
|
|
|
* Flash memory and Backup registers as described for Level 1. However,
|
|
|
|
|
* there are three major differences. The JTAG/SWD debugger connection is
|
|
|
|
|
* disabled (even at the ST factory, to ensure that there are no
|
|
|
|
|
* backdoors), the Boot mode is forced to User Flash memory REGARDLESS of
|
|
|
|
|
* what the boot 0/1 settings are, and Level 2 is permanent. Once set to
|
|
|
|
|
* Level 2, there is no going back; RDP/WRP option bytes can no longer be
|
|
|
|
|
* changed, as well as ALL the other option bytes.
|
|
|
|
|
*
|
|
|
|
|
* By way of background, changing the level of RDP protection is only
|
|
|
|
|
* permitted when the current protection level is ‘1’. Changing the protection
|
|
|
|
|
* level from '1' to '0' should automatically erase the entire user flash
|
|
|
|
|
* memory, SRAM2 and backup registers.
|
|
|
|
|
*
|
|
|
|
|
* The issue is that while Level 0 is 0xAA and Level 2 is 0xCC, Level 1 is any
|
|
|
|
|
* other number. So when OxCC is set and the chip is physically or
|
|
|
|
|
* electrically perturbed, flipping any bit will "fool" the CPU into thinking
|
|
|
|
|
* that it is in Level 1, allowing JTAG access and the changing of option
|
|
|
|
|
* bits.
|
|
|
|
|
*
|
|
|
|
|
* Think of this as a STM32-specific version of the Rowhammer attack.
|
|
|
|
|
*
|
|
|
|
|
* RDP may not be set correctly due to manufacturing error, glitch or
|
2023-10-16 12:17:48 +02:00
|
|
|
|
* intentional attack. It's done thrice to reduce the probability of a
|
2021-10-12 20:00:19 +02:00
|
|
|
|
* glitch attack succeeding amongst all of the multireads desgned to make it
|
|
|
|
|
* tougher.
|
|
|
|
|
*
|
|
|
|
|
* This would be best served with a random delay at the beginning of the
|
|
|
|
|
* function. But a consistent strategy for all chips is tough.
|
|
|
|
|
*
|
|
|
|
|
* To set the RDP bytes, the J-Flash utility or the STM32 Unlock (from J-Link)
|
|
|
|
|
* utility, both provided by the manufacturer.
|
|
|
|
|
*
|
|
|
|
|
* You can also set the option bytes from code:
|
|
|
|
|
*
|
|
|
|
|
* 1. Unlock the option bytes by writing the correct keys to FLASH_OPTKEYR and
|
|
|
|
|
* clearing OPTLOCK
|
|
|
|
|
* 2. Set the desired option values in FLASH_OPTCR
|
|
|
|
|
* 3. Set OPTSTRT in FLASH_OPTCR
|
|
|
|
|
*
|
|
|
|
|
* This is the generic procedure for all option bytes. However, setting the
|
|
|
|
|
* RDP level in this fashion will immediately lock the CPU and force a reboot
|
|
|
|
|
* (and in some cases a clearing of the flash memory).
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* RDP only defined for particular families. Kconfig sets this as necessary */
|
|
|
|
|
#if defined(STM32_OPTION_BYTES)
|
|
|
|
|
|
|
|
|
|
#ifndef CONFIG_STM32_RDP
|
|
|
|
|
#define CONFIG_STM32_RDP 0
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
static bool _rdp_ok(void)
|
|
|
|
|
{
|
|
|
|
|
if (CONFIG_STM32_RDP == 0) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
bool glitch = false;
|
|
|
|
|
uint32_t read1 = _multi_read_reg32(STM32_OPTION_BYTES, &glitch);
|
|
|
|
|
uint32_t read2 = _multi_read_reg32(STM32_OPTION_BYTES, &glitch);
|
|
|
|
|
uint32_t read3 = _multi_read_reg32(STM32_OPTION_BYTES, &glitch);
|
|
|
|
|
if (glitch) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (CONFIG_STM32_RDP) {
|
|
|
|
|
case 1:
|
|
|
|
|
return GET_RDP(read1) == 0xAA ||
|
|
|
|
|
GET_RDP(read2) == 0xAA ||
|
|
|
|
|
GET_RDP(read3) == 0xAA;
|
|
|
|
|
case 2:
|
|
|
|
|
return GET_RDP(read1) != 0xCC ||
|
|
|
|
|
GET_RDP(read2) != 0xCC ||
|
|
|
|
|
GET_RDP(read3) != 0xCC;
|
|
|
|
|
default:
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void _rdp_check(void)
|
|
|
|
|
{
|
|
|
|
|
if (!_rdp_ok()) {
|
|
|
|
|
/* halt execution */
|
|
|
|
|
while (1) {
|
|
|
|
|
pm_set(0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif /* STM32_OPTION_BYTES */
|
|
|
|
|
|
2021-07-07 13:41:56 +02:00
|
|
|
|
/**
|
|
|
|
|
* @brief Initialize HW debug pins for Sub-GHz Radio
|
|
|
|
|
*/
|
2021-07-21 11:26:17 +02:00
|
|
|
|
void _wlx5xx_init_subghz_debug_pins(void)
|
2021-07-07 13:41:56 +02:00
|
|
|
|
{
|
2021-07-21 11:26:17 +02:00
|
|
|
|
#if IS_ACTIVE(CONFIG_STM32_WLX5XX_SUBGHZ_DEBUG)
|
2021-07-07 13:41:56 +02:00
|
|
|
|
/* SUBGHZSPI Debug */
|
|
|
|
|
gpio_init(CPU_STM32WL_SUBGHZSPI_DEBUG_MOSIOUT, GPIO_OUT);
|
|
|
|
|
gpio_init_af(CPU_STM32WL_SUBGHZSPI_DEBUG_MOSIOUT,
|
|
|
|
|
CPU_STM32WL_SUBGHZSPI_DEBUG_MOSIOUT_AF);
|
|
|
|
|
|
|
|
|
|
gpio_init(CPU_STM32WL_SUBGHZSPI_DEBUG_MISOOUT, GPIO_OUT);
|
|
|
|
|
gpio_init_af(CPU_STM32WL_SUBGHZSPI_DEBUG_MISOOUT,
|
|
|
|
|
CPU_STM32WL_SUBGHZSPI_DEBUG_MISOOUT_AF);
|
|
|
|
|
|
|
|
|
|
gpio_init(CPU_STM32WL_SUBGHZSPI_DEBUG_SCKOUT, GPIO_OUT);
|
|
|
|
|
gpio_init_af(CPU_STM32WL_SUBGHZSPI_DEBUG_SCKOUT,
|
|
|
|
|
CPU_STM32WL_SUBGHZSPI_DEBUG_SCKOUT_AF);
|
|
|
|
|
|
|
|
|
|
gpio_init(CPU_STM32WL_SUBGHZSPI_DEBUG_NSSOUT, GPIO_OUT);
|
|
|
|
|
gpio_init_af(CPU_STM32WL_SUBGHZSPI_DEBUG_NSSOUT,
|
|
|
|
|
CPU_STM32WL_SUBGHZSPI_DEBUG_NSSOUT_AF);
|
|
|
|
|
|
|
|
|
|
/* Sub-GHz Radio Debug */
|
|
|
|
|
gpio_init(CPU_STM32WL_SUBGHZ_RF_BUSY, GPIO_OUT);
|
|
|
|
|
gpio_init_af(CPU_STM32WL_SUBGHZ_RF_BUSY,
|
|
|
|
|
CPU_STM32WL_SUBGHZ_RF_BUSY_AF);
|
|
|
|
|
|
|
|
|
|
gpio_init(CPU_STM32WL_SUBGHZ_DEBUG_RF_NRESET, GPIO_OUT);
|
|
|
|
|
gpio_init_af(CPU_STM32WL_SUBGHZ_DEBUG_RF_NRESET,
|
|
|
|
|
CPU_STM32WL_SUBGHZ_DEBUG_RF_NRESET_AF);
|
|
|
|
|
|
|
|
|
|
gpio_init(CPU_STM32WL_SUBGHZ_DEBUG_RF_SMPSRDY, GPIO_OUT);
|
|
|
|
|
gpio_init_af(CPU_STM32WL_SUBGHZ_DEBUG_RF_SMPSRDY,
|
|
|
|
|
CPU_STM32WL_SUBGHZ_DEBUG_RF_SMPSRDY_AF);
|
|
|
|
|
|
|
|
|
|
gpio_init(CPU_STM32WL_SUBGHZ_DEBUG_RF_LDORDY, GPIO_OUT);
|
|
|
|
|
gpio_init_af(CPU_STM32WL_SUBGHZ_DEBUG_RF_LDORDY,
|
|
|
|
|
CPU_STM32WL_SUBGHZ_DEBUG_RF_LDORDY_AF);
|
|
|
|
|
|
|
|
|
|
gpio_init(CPU_STM32WL_SUBGHZ_DEBUG_RF_HSE32RDY, GPIO_OUT);
|
|
|
|
|
gpio_init_af(CPU_STM32WL_SUBGHZ_DEBUG_RF_HSE32RDY,
|
|
|
|
|
CPU_STM32WL_SUBGHZ_DEBUG_RF_HSE32RDY_AF);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-04 06:49:29 +02:00
|
|
|
|
static void swj_init(void)
|
|
|
|
|
{
|
|
|
|
|
#if defined(CPU_FAM_STM32F1)
|
|
|
|
|
/* Only if the selected SWJ config differs from the reset value, we
|
|
|
|
|
* actually need to do something. Since both sides are compile time
|
|
|
|
|
* constants, this hole code gets optimized out by default */
|
|
|
|
|
if (CONFIG_AFIO_MAPR_SWJ_CFG != SWJ_CFG_FULL_SWJ) {
|
|
|
|
|
/* The remapping periph clock must first be enabled */
|
|
|
|
|
RCC->APB2ENR |= RCC_APB2ENR_AFIOEN;
|
|
|
|
|
/* Handling of the MAPR register is a bit involved due to the
|
|
|
|
|
* write-only nature of the SWJ_CFG field, which returns undefined
|
|
|
|
|
* garbage on read. `afio_mapr_read()` will read the current MAPR
|
|
|
|
|
* value, but clear the SWF_CFG vield. `afio_mapr_wriote()` will then
|
|
|
|
|
* write the value read back, but apply the `SWF_CFG` configuration
|
|
|
|
|
* from `CONFIG_AFIO_MAPR_SWJ_CFG` first.*/
|
|
|
|
|
afio_mapr_write(afio_mapr_read());
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-03 14:35:01 +02:00
|
|
|
|
void cpu_init(void)
|
|
|
|
|
{
|
|
|
|
|
/* initialize the Cortex-M core */
|
|
|
|
|
cortexm_init();
|
|
|
|
|
/* enable PWR module */
|
2021-12-15 15:53:53 +01:00
|
|
|
|
#if defined(CPU_FAM_STM32U5)
|
|
|
|
|
periph_clk_en(AHB3, RCC_AHB3ENR_PWREN);
|
|
|
|
|
#elif !defined(CPU_FAM_STM32WB) && !defined(CPU_FAM_STM32MP1) && \
|
2021-03-05 01:00:00 +01:00
|
|
|
|
!defined(CPU_FAM_STM32WL)
|
2020-05-03 14:35:01 +02:00
|
|
|
|
periph_clk_en(APB1, BIT_APB_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)
|
|
|
|
|
_gpio_init_ain();
|
2021-10-12 20:00:19 +02:00
|
|
|
|
_rdp_check();
|
2020-05-03 14:35:01 +02:00
|
|
|
|
#endif
|
2020-09-05 01:33:45 +02:00
|
|
|
|
#if !defined(CPU_FAM_STM32MP1) || IS_USED(MODULE_STM32MP1_ENG_MODE)
|
2020-09-24 11:25:30 +02:00
|
|
|
|
/* initialize the system clock as configured in the periph_conf.h */
|
|
|
|
|
stmclk_init_sysclk();
|
2020-09-05 01:33:45 +02:00
|
|
|
|
#endif
|
2020-05-03 14:35:01 +02:00
|
|
|
|
#ifdef MODULE_PERIPH_DMA
|
|
|
|
|
/* initialize DMA streams */
|
|
|
|
|
dma_init();
|
|
|
|
|
#endif
|
|
|
|
|
/* initialize stdio prior to periph_init() to allow use of DEBUG() there */
|
2023-01-02 18:08:35 +01:00
|
|
|
|
early_init();
|
2020-05-03 14:35:01 +02:00
|
|
|
|
|
2023-07-04 06:49:29 +02:00
|
|
|
|
swj_init();
|
2020-05-03 14:35:01 +02:00
|
|
|
|
|
|
|
|
|
/* trigger static peripheral initialization */
|
|
|
|
|
periph_init();
|
2021-07-07 13:41:56 +02:00
|
|
|
|
|
2021-07-21 11:26:17 +02:00
|
|
|
|
if (IS_ACTIVE(CONFIG_STM32_WLX5XX_SUBGHZ_DEBUG)) {
|
|
|
|
|
_wlx5xx_init_subghz_debug_pins();
|
2021-07-07 13:41:56 +02:00
|
|
|
|
}
|
2020-05-03 14:35:01 +02:00
|
|
|
|
}
|
2021-09-16 09:58:37 +02:00
|
|
|
|
|
|
|
|
|
void backup_ram_init(void)
|
|
|
|
|
{
|
|
|
|
|
/* see reference manual "Battery backup domain" */
|
|
|
|
|
#if defined(RCC_APB1ENR_PWREN)
|
|
|
|
|
periph_clk_en(APB1, RCC_APB1ENR_PWREN);
|
|
|
|
|
#elif defined(RCC_APBENR1_PWREN)
|
|
|
|
|
periph_clk_en(APB1, RCC_APBENR1_PWREN);
|
|
|
|
|
#elif defined(RCC_APB1ENR1_PWREN)
|
|
|
|
|
periph_clk_en(APB1, RCC_APB1ENR1_PWREN);
|
|
|
|
|
#elif defined(RCC_AHB3ENR_PWREN)
|
|
|
|
|
periph_clk_en(AHB3, RCC_AHB3ENR_PWREN);
|
|
|
|
|
#endif
|
|
|
|
|
stmclk_dbp_unlock();
|
|
|
|
|
#if defined(RCC_AHB1ENR_BKPSRAMEN)
|
|
|
|
|
periph_clk_en(AHB1, RCC_AHB1ENR_BKPSRAMEN);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifndef BACKUP_RAM_MAGIC
|
|
|
|
|
#define BACKUP_RAM_MAGIC {'R', 'I', 'O', 'T'}
|
|
|
|
|
#endif
|
|
|
|
|
|
2023-05-19 22:09:01 +02:00
|
|
|
|
__attribute__((unused))
|
2021-10-07 21:08:27 +02:00
|
|
|
|
static inline bool _backup_battery_connected(void) {
|
|
|
|
|
#if IS_USED(MODULE_PERIPH_VBAT)
|
|
|
|
|
vbat_init(); /* early use of VBAT requires init() */
|
|
|
|
|
return !vbat_is_empty();
|
|
|
|
|
#endif
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool cpu_woke_from_backup(void) {
|
2021-09-16 09:58:37 +02:00
|
|
|
|
#if IS_ACTIVE(CPU_HAS_BACKUP_RAM)
|
|
|
|
|
static const char _signature[] BACKUP_RAM_DATA = BACKUP_RAM_MAGIC;
|
2021-10-07 21:08:27 +02:00
|
|
|
|
if (_backup_battery_connected()) {
|
|
|
|
|
/* in case the board has a backup battery the regulator must be on
|
|
|
|
|
to mitigate (unexpected) outage of VDD, so RTC register and
|
|
|
|
|
backup domain register contents are not lost */
|
|
|
|
|
pm_backup_regulator_on();
|
|
|
|
|
}
|
|
|
|
|
else {
|
2021-09-16 09:58:37 +02:00
|
|
|
|
#ifndef RIOTBOOT
|
2021-10-07 21:08:27 +02:00
|
|
|
|
/* switch off regulator to save power */
|
|
|
|
|
pm_backup_regulator_off();
|
2021-09-16 09:58:37 +02:00
|
|
|
|
#endif
|
2021-10-07 21:08:27 +02:00
|
|
|
|
}
|
2021-09-16 09:58:37 +02:00
|
|
|
|
for (unsigned i = 0; i < sizeof(_signature); i++) {
|
|
|
|
|
if (_signature[i] != ((char[])BACKUP_RAM_MAGIC)[i]) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
#endif /* CPU_HAS_BACKUP_RAM */
|
|
|
|
|
return false;
|
|
|
|
|
}
|