2018-06-11 06:34:03 -07:00
|
|
|
/************************************************************************************//**
|
2018-07-17 06:06:06 -07:00
|
|
|
* \file Demo/ARMCM4_STM32F3_Discovery_F303VC_GCC/Boot/main.c
|
2018-06-11 06:34:03 -07:00
|
|
|
* \brief Bootloader application source file.
|
|
|
|
* \ingroup Boot_ARMCM4_STM32F3_Discovery_F303VC_GCC
|
|
|
|
* \internal
|
|
|
|
*----------------------------------------------------------------------------------------
|
|
|
|
* C O P Y R I G H T
|
|
|
|
*----------------------------------------------------------------------------------------
|
|
|
|
* Copyright (c) 2018 by Feaser http://www.feaser.com All rights reserved
|
|
|
|
*
|
|
|
|
*----------------------------------------------------------------------------------------
|
|
|
|
* L I C E N S E
|
|
|
|
*----------------------------------------------------------------------------------------
|
|
|
|
* This file is part of OpenBLT. OpenBLT is free software: you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as published by the Free
|
|
|
|
* Software Foundation, either version 3 of the License, or (at your option) any later
|
|
|
|
* version.
|
|
|
|
*
|
|
|
|
* OpenBLT is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
|
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
|
|
* PURPOSE. See the GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You have received a copy of the GNU General Public License along with OpenBLT. It
|
|
|
|
* should be located in ".\Doc\license.html". If not, contact Feaser to obtain a copy.
|
|
|
|
*
|
|
|
|
* \endinternal
|
|
|
|
****************************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************************
|
|
|
|
* Include files
|
|
|
|
****************************************************************************************/
|
|
|
|
#include "boot.h" /* bootloader generic header */
|
|
|
|
#include "stm32f3xx.h" /* STM32 CPU and HAL header */
|
|
|
|
#include "stm32f3xx_ll_rcc.h" /* STM32 LL RCC header */
|
|
|
|
#include "stm32f3xx_ll_bus.h" /* STM32 LL BUS header */
|
|
|
|
#include "stm32f3xx_ll_system.h" /* STM32 LL SYSTEM header */
|
|
|
|
#include "stm32f3xx_ll_utils.h" /* STM32 LL UTILS header */
|
|
|
|
#include "stm32f3xx_ll_usart.h" /* STM32 LL USART header */
|
|
|
|
#include "stm32f3xx_ll_gpio.h" /* STM32 LL GPIO header */
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************************
|
|
|
|
* Function prototypes
|
|
|
|
****************************************************************************************/
|
|
|
|
static void Init(void);
|
|
|
|
static void SystemClock_Config(void);
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************************************************//**
|
|
|
|
** \brief This is the entry point for the bootloader application and is called
|
|
|
|
** by the reset interrupt vector after the C-startup routines executed.
|
|
|
|
** \return Program return code.
|
|
|
|
**
|
|
|
|
****************************************************************************************/
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
/* initialize the microcontroller */
|
|
|
|
Init();
|
|
|
|
/* initialize the bootloader */
|
|
|
|
BootInit();
|
|
|
|
|
|
|
|
/* start the infinite program loop */
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
/* run the bootloader task */
|
|
|
|
BootTask();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* program should never get here */
|
|
|
|
return 0;
|
|
|
|
} /*** end of main ***/
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************************************************//**
|
|
|
|
** \brief Initializes the microcontroller.
|
|
|
|
** \return none.
|
|
|
|
**
|
|
|
|
****************************************************************************************/
|
|
|
|
static void Init(void)
|
|
|
|
{
|
|
|
|
/* HAL library initialization */
|
|
|
|
HAL_Init();
|
|
|
|
/* configure system clock */
|
|
|
|
SystemClock_Config();
|
|
|
|
} /*** end of Init ***/
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************************************************//**
|
|
|
|
** \brief System Clock Configuration. This code was created by CubeMX and configures
|
|
|
|
** the system clock to match the configuration in the bootloader's
|
|
|
|
** configuration (blt_conf.h), specifically the macros:
|
|
|
|
** BOOT_CPU_SYSTEM_SPEED_KHZ and BOOT_CPU_XTAL_SPEED_KHZ.
|
|
|
|
** Note that the Lower Layer drivers were selected in CubeMX for the RCC
|
|
|
|
** subsystem.
|
|
|
|
** \return none.
|
|
|
|
**
|
|
|
|
****************************************************************************************/
|
|
|
|
static void SystemClock_Config(void)
|
|
|
|
{
|
|
|
|
/* Set flash latency. */
|
|
|
|
LL_FLASH_SetLatency(LL_FLASH_LATENCY_2);
|
|
|
|
/* Verify flash latency setting. */
|
|
|
|
if(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_2)
|
|
|
|
{
|
|
|
|
/* Error setting flash latency. */
|
|
|
|
ASSERT_RT(BLT_FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enable the HSE bypass clock. */
|
|
|
|
LL_RCC_HSE_EnableBypass();
|
|
|
|
LL_RCC_HSE_Enable();
|
|
|
|
|
|
|
|
/* Wait till HSE is ready */
|
|
|
|
while(LL_RCC_HSE_IsReady() != 1)
|
|
|
|
{
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Configure and enable the PLL. */
|
|
|
|
LL_RCC_PLL_ConfigDomain_SYS(LL_RCC_PLLSOURCE_HSE_DIV_1, LL_RCC_PLL_MUL_9);
|
|
|
|
LL_RCC_PLL_Enable();
|
|
|
|
/* Wait till PLL is ready */
|
|
|
|
while(LL_RCC_PLL_IsReady() != 1)
|
|
|
|
{
|
|
|
|
;
|
|
|
|
}
|
|
|
|
LL_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_1);
|
|
|
|
LL_RCC_SetAPB1Prescaler(LL_RCC_APB1_DIV_2);
|
|
|
|
LL_RCC_SetAPB2Prescaler(LL_RCC_APB1_DIV_1);
|
|
|
|
LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_PLL);
|
|
|
|
/* Wait till System clock is ready */
|
|
|
|
while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_PLL)
|
|
|
|
{
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if (BOOT_COM_USB_ENABLE > 0)
|
|
|
|
LL_RCC_SetUSBClockSource(LL_RCC_USB_CLKSOURCE_PLL_DIV_1_5);
|
|
|
|
#endif
|
|
|
|
/* Update the system clock speed setting. */
|
|
|
|
LL_SetSystemCoreClock(BOOT_CPU_SYSTEM_SPEED_KHZ * 1000u);
|
|
|
|
} /*** end of SystemClock_Config ***/
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************************************************//**
|
|
|
|
** \brief Initializes the Global MSP. This function is called from HAL_Init()
|
|
|
|
** function to perform system level initialization (GPIOs, clock, DMA,
|
|
|
|
** interrupt).
|
|
|
|
** \return none.
|
|
|
|
**
|
|
|
|
****************************************************************************************/
|
|
|
|
void HAL_MspInit(void)
|
|
|
|
{
|
|
|
|
LL_GPIO_InitTypeDef GPIO_InitStruct;
|
|
|
|
|
|
|
|
/* SYSCFG clock enable. */
|
|
|
|
LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_SYSCFG);
|
|
|
|
|
|
|
|
/* GPIO ports clock enable. */
|
|
|
|
LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_GPIOF);
|
|
|
|
LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_GPIOA);
|
|
|
|
LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_GPIOE);
|
|
|
|
|
2020-02-06 08:22:58 -08:00
|
|
|
#if (BOOT_COM_RS232_ENABLE > 0)
|
2018-06-11 06:34:03 -07:00
|
|
|
/* UART clock enable. */
|
|
|
|
LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_USART2);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Configure GPIO pin for the LED. */
|
|
|
|
GPIO_InitStruct.Pin = LL_GPIO_PIN_8;
|
|
|
|
GPIO_InitStruct.Mode = LL_GPIO_MODE_OUTPUT;
|
|
|
|
GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_LOW;
|
|
|
|
GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
|
|
|
|
GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
|
|
|
|
LL_GPIO_Init(GPIOE, &GPIO_InitStruct);
|
|
|
|
LL_GPIO_ResetOutputPin(GPIOE, LL_GPIO_PIN_8);
|
|
|
|
|
|
|
|
/* Configure GPIO pin for (optional) backdoor entry input. */
|
|
|
|
GPIO_InitStruct.Pin = LL_GPIO_PIN_0;
|
|
|
|
GPIO_InitStruct.Mode = LL_GPIO_MODE_INPUT;
|
|
|
|
GPIO_InitStruct.Pull = LL_GPIO_PULL_DOWN;
|
|
|
|
LL_GPIO_Init(GPIOA, &GPIO_InitStruct);
|
|
|
|
|
2020-02-06 08:22:58 -08:00
|
|
|
#if (BOOT_COM_RS232_ENABLE > 0)
|
2018-06-11 06:34:03 -07:00
|
|
|
/* UART TX and RX GPIO pin configuration. */
|
|
|
|
GPIO_InitStruct.Pin = LL_GPIO_PIN_2|LL_GPIO_PIN_3;
|
|
|
|
GPIO_InitStruct.Mode = LL_GPIO_MODE_ALTERNATE;
|
|
|
|
GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_HIGH;
|
|
|
|
GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
|
|
|
|
GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
|
|
|
|
GPIO_InitStruct.Alternate = LL_GPIO_AF_7;
|
|
|
|
LL_GPIO_Init(GPIOA, &GPIO_InitStruct);
|
|
|
|
#endif
|
|
|
|
} /*** end of HAL_MspInit ***/
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************************************************//**
|
|
|
|
** \brief DeInitializes the Global MSP. This function is called from HAL_DeInit()
|
|
|
|
** function to perform system level de-initialization (GPIOs, clock, DMA,
|
|
|
|
** interrupt).
|
|
|
|
** \return none.
|
|
|
|
**
|
|
|
|
****************************************************************************************/
|
|
|
|
void HAL_MspDeInit(void)
|
|
|
|
{
|
2023-06-16 03:03:44 -07:00
|
|
|
LL_GPIO_InitTypeDef GPIO_InitStruct = {0};
|
|
|
|
|
|
|
|
/* The STM32F3-Discovery board has a pull-up on the USB_DP line, which is always
|
|
|
|
* enabled by default. If the USB USER cable is connected (for example after a
|
|
|
|
* firmware update with the bootloader), this pull-up causes the USB host to try
|
|
|
|
* and enumerate the USB device.
|
|
|
|
* This enumeration will fail if the user program itself does not make use of the USB
|
|
|
|
* peripheral. This failed enumeration is not a problem for the user program, but might
|
|
|
|
* cause the bootloader to not enumerate properly after starting it via a system reset.
|
|
|
|
* To prevent enumeration by the USB host, place the USB device in a disconnected
|
|
|
|
* state, which is done by configuring the USB_DP line as a digital output and setting
|
|
|
|
* it to logic low.
|
|
|
|
*/
|
|
|
|
LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_GPIOA);
|
|
|
|
LL_GPIO_ResetOutputPin(GPIOA, GPIO_PIN_12);
|
|
|
|
GPIO_InitStruct.Pin = GPIO_PIN_12;
|
|
|
|
GPIO_InitStruct.Mode = LL_GPIO_MODE_OUTPUT;
|
|
|
|
GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_LOW;
|
|
|
|
GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
|
|
|
|
GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
|
|
|
|
LL_GPIO_Init(GPIOA, &GPIO_InitStruct);
|
|
|
|
|
2020-02-15 02:54:43 -08:00
|
|
|
/* Reset the RCC clock configuration to the default reset state. */
|
|
|
|
LL_RCC_DeInit();
|
|
|
|
|
2018-06-11 06:34:03 -07:00
|
|
|
/* Reset GPIO pin for the LED to turn it off. */
|
|
|
|
LL_GPIO_ResetOutputPin(GPIOE, LL_GPIO_PIN_8);
|
|
|
|
|
2023-06-16 03:03:44 -07:00
|
|
|
/* Deinit used GPIOs, except GPIOA to make sure USB D+ (PA12) stays low. */
|
2018-06-11 06:34:03 -07:00
|
|
|
LL_GPIO_DeInit(GPIOE);
|
|
|
|
|
2020-02-06 08:22:58 -08:00
|
|
|
#if (BOOT_COM_RS232_ENABLE > 0)
|
2018-06-11 06:34:03 -07:00
|
|
|
/* UART clock disable. */
|
|
|
|
LL_APB1_GRP1_DisableClock(LL_APB1_GRP1_PERIPH_USART2);
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2023-06-16 03:03:44 -07:00
|
|
|
/* GPIO ports clock disable, except GPIOA to make sure USB D+ (PA12) stays low. */
|
2018-06-11 06:34:03 -07:00
|
|
|
LL_AHB1_GRP1_DisableClock(LL_AHB1_GRP1_PERIPH_GPIOE);
|
|
|
|
LL_AHB1_GRP1_DisableClock(LL_AHB1_GRP1_PERIPH_GPIOF);
|
|
|
|
|
|
|
|
/* SYSCFG clock disable. */
|
|
|
|
LL_APB2_GRP1_DisableClock(LL_APB2_GRP1_PERIPH_SYSCFG);
|
|
|
|
} /*** end of HAL_MspDeInit ***/
|
|
|
|
|
|
|
|
|
2022-08-08 06:28:31 -07:00
|
|
|
/************************************************************************************//**
|
|
|
|
** \brief This function handles the SysTick interrupt. The HAL driver is initialized
|
|
|
|
** before the bootloader disables the global interrupts and reconfigures the
|
|
|
|
** SysTick. It is theoretically possible that the SysTick interrupt still
|
|
|
|
** fires before the timer driver disables it. Therefore the handler is
|
|
|
|
** implemented here. If not, then the default handler from the C startup
|
|
|
|
** code is used, which hangs the system.
|
|
|
|
** \return none.
|
|
|
|
**
|
|
|
|
****************************************************************************************/
|
|
|
|
void SysTick_Handler(void)
|
|
|
|
{
|
|
|
|
/* Nothing to do here. */
|
|
|
|
} /*** end of SysTick_Handler ***/
|
|
|
|
|
|
|
|
|
2018-06-11 06:34:03 -07:00
|
|
|
/*********************************** end of main.c *************************************/
|