stm32l4: add support for booting into DFU mode

This commit is contained in:
Matias Nitsche 2020-06-15 12:34:42 -03:00 committed by Alan Carvalho de Assis
parent b83b83bf51
commit b4bea95a6a
3 changed files with 230 additions and 0 deletions

View File

@ -100,6 +100,7 @@ CHIP_CSRCS += stm32l4_irq.c stm32l4_lowputc.c stm32l4_rcc.c
CHIP_CSRCS += stm32l4_serial.c stm32l4_start.c stm32l4_waste.c stm32l4_uid.c CHIP_CSRCS += stm32l4_serial.c stm32l4_start.c stm32l4_waste.c stm32l4_uid.c
CHIP_CSRCS += stm32l4_spi.c stm32l4_i2c.c stm32l4_lse.c stm32l4_lsi.c CHIP_CSRCS += stm32l4_spi.c stm32l4_i2c.c stm32l4_lse.c stm32l4_lsi.c
CHIP_CSRCS += stm32l4_pwr.c stm32l4_tim.c stm32l4_flash.c CHIP_CSRCS += stm32l4_pwr.c stm32l4_tim.c stm32l4_flash.c
CHIP_CSRCS += stm32l4_dfumode.c
ifneq ($(CONFIG_ARCH_IDLE_CUSTOM),y) ifneq ($(CONFIG_ARCH_IDLE_CUSTOM),y)
CHIP_CSRCS += stm32l4_idle.c CHIP_CSRCS += stm32l4_idle.c

View File

@ -0,0 +1,168 @@
/****************************************************************************
* arch/arm/src/stm32/stm32l4_dfumode.c
*
* Copyright (C) 2019 Bill Gatliff. All rights reserved.
* Copyright (C) 2019 Gregory Nutt. All rights reserved.
* Author: Bill Gatliff <bgat@billgatliff.com>
* Gregory Nutt <gnutt@nuttx.org>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* 3. Neither the name NuttX nor the names of its contributors may be
* used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <unistd.h>
#include <debug.h>
#include <nuttx/signal.h>
#include "stm32l4_rcc.h"
#include "hardware/stm32l4_syscfg.h"
#include "stm32l4_dfumode.h"
static inline void rcc_reset(void)
{
uint32_t regval;
regval = getreg32(STM32L4_RCC_CR);
regval |= RCC_CR_MSION;
putreg32(regval, STM32L4_RCC_CR);
while (!(getreg32(STM32L4_RCC_CR) & RCC_CR_MSIRDY));
/* Set MSI to 4MHz */
regval = getreg32(STM32L4_RCC_CR);
regval &= ~RCC_CR_MSIRANGE_MASK;
regval |= RCC_CR_MSIRANGE_4M | RCC_CR_MSIRGSEL;
putreg32(regval, STM32L4_RCC_CR);
/* Reset CFGR register */
putreg32(0x00000000, STM32L4_RCC_CFGR);
/* Reset HSION, HSEON, CSSON and PLLON bits */
regval = getreg32(STM32L4_RCC_CR);
regval &= ~(RCC_CR_HSION | RCC_CR_HSIKERON | RCC_CR_HSEON | RCC_CR_HSIASFS |
RCC_CR_CSSON | RCC_CR_PLLON | RCC_CR_PLLON | RCC_CR_PLLSAI1ON |
RCC_CR_PLLSAI2ON);
putreg32(regval, STM32L4_RCC_CR);
/* Reset PLLCFGR register to reset default */
putreg32(RCC_PLLCFG_RESET, STM32L4_RCC_PLLCFG);
putreg32(0, STM32L4_RCC_PLLSAI1CFG);
putreg32(RCC_PLLSAI1CFG_PLLN(16), STM32L4_RCC_PLLSAI1CFG);
putreg32(0, STM32L4_RCC_PLLSAI2CFG);
putreg32(RCC_PLLSAI2CFG_PLLN(16), STM32L4_RCC_PLLSAI1CFG);
/* Reset HSEBYP bit */
regval = getreg32(STM32L4_RCC_CR);
regval &= ~RCC_CR_HSEBYP;
putreg32(regval, STM32L4_RCC_CR);
/* Disable all interrupts */
putreg32(0x00000000, STM32L4_RCC_CIER);
}
static inline void apb_reset(void)
{
putreg32(0xffffffff, STM32L4_RCC_APB1RSTR1);
putreg32(0xffffffff, STM32L4_RCC_APB1RSTR2);
putreg32(0xffffffff, STM32L4_RCC_APB2RSTR);
putreg32(0xffffffff, STM32L4_RCC_AHB1RSTR);
putreg32(0xffffffff, STM32L4_RCC_AHB2RSTR);
putreg32(0xffffffff, STM32L4_RCC_AHB3RSTR);
putreg32(0, STM32L4_RCC_APB1RSTR1);
putreg32(0, STM32L4_RCC_APB1RSTR2);
putreg32(0, STM32L4_RCC_APB2RSTR);
putreg32(0, STM32L4_RCC_AHB1RSTR);
putreg32(0, STM32L4_RCC_AHB2RSTR);
putreg32(0, STM32L4_RCC_AHB3RSTR);
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: stm32l4_dfumode
*
* Description:
* Reboot the part in DFU mode (GCC only).
*
****************************************************************************/
#if defined(CONFIG_STM32L4_STM32L4X6)
void stm32l4_dfumode(void)
{
uint32_t regval;
void (*SysMemBootJump)(void);
#ifdef CONFIG_DEBUG_WARN
_warn("Entering DFU mode...\n");
#endif
/* disable all peripherals, interrupts and switch to HSI */
rcc_reset();
apb_reset();
/* remap ROM at address zero */
regval = getreg32(STM32L4_RCC_APB2ENR);
regval |= RCC_APB2ENR_SYSCFGEN;
putreg32(regval, STM32L4_RCC_APB2ENR);
putreg32(SYSCFG_MEMRMP_SYSTEM, STM32L4_SYSCFG_MEMRMP);
/* set stack pointer and program-counter to ROM values */
asm("dsb\n\t"
"isb\n\t"
"ldr r0, =0x1fff0000\n\t" /* ROM base */
"ldr sp,[r0, #0]\n\t" /* SP @ 0 */
);
/* jump into ROM */
SysMemBootJump = (void (*)(void))(*((uint32_t*)0x1fff0004));
SysMemBootJump();
while(1);
__builtin_unreachable(); /* Tell compiler we will not return */
}
#endif

View File

@ -0,0 +1,61 @@
/****************************************************************************
* arch/arm/src/stm32/stm32l4_dfumode.h
*
* Copyright (C) 2019 Gregory Nutt. All rights reserved.
* Author: Gregory Nutt <gnutt@nuttx.org>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* 3. Neither the name NuttX nor the names of its contributors may be
* used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
****************************************************************************/
#ifndef __ARCH_ARM_SRC_STM32L4_DFUMODE_H
#define __ARCH_ARM_SRC_STM32L4_DFUMODE_H
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/compiler.h>
/****************************************************************************
* Public Function Prototypes
****************************************************************************/
/****************************************************************************
* Name: stm32l4_dfumode
*
* Description:
* Reboot the part in DFU mode.
*
****************************************************************************/
#if defined(CONFIG_STM32L4_STM32L4X6)
void stm32l4_dfumode(void) noreturn_function;
#endif
#endif /* __ARCH_ARM_SRC_STM32L4_DFUMODE_H */