HAL库U盘升级 STM32F407 CUBEMX:FATFS + USB_HOST + USB_OTG_FS

一、测试平台:

MCU:STM32F407VET6
固件库:CUBEMX
IDE:MDK

二、实验目的:

将U盘里面的bin文件插入要升级的设备,通过BootLoader来进行升级

在这是用板载的LED灯来显示升级情况:
不进行升级:LED灯是灭的状态
升级成功:LED灯以100ms在闪烁

接下来先进行BootLoader的配置以及程序编写,再配置APP

三、BootLoader:

1 下载器配置请添加图片描述
2 时钟源配置请添加图片描述
3 LED配置请添加图片描述

4 串口配置 开启全局中断请添加图片描述
5 USB_OTG_FS配置请添加图片描述
6 USB_HOST配置请添加图片描述

7 FATFS配置请添加图片描述

8 时钟树配置 48M是用来操作U盘的,所以必须要有请添加图片描述

9 工程配置请添加图片描述

10 生成工程请添加图片描述

11 创建两个空白文件文件请添加图片描述

BootLoader.h

#ifndef __BOOTLOADER_H
#define __BOOTLOADER_H

#include "stm32f4xx_hal.h"
#include "usb_host.h"
#include "fatfs.h"
#include "usart.h"

#define ADDR_FLASH_SECTOR_0     ((uint32_t)0x08000000) /* Base @ of Sector 0, 16 Kbytes */
#define ADDR_FLASH_SECTOR_1     ((uint32_t)0x08004000) /* Base @ of Sector 1, 16 Kbytes */
#define ADDR_FLASH_SECTOR_2     ((uint32_t)0x08008000) /* Base @ of Sector 2, 16 Kbytes */
#define ADDR_FLASH_SECTOR_3     ((uint32_t)0x0800C000) /* Base @ of Sector 3, 16 Kbytes */
#define ADDR_FLASH_SECTOR_4     ((uint32_t)0x08010000) /* Base @ of Sector 4, 64 Kbytes */
#define ADDR_FLASH_SECTOR_5     ((uint32_t)0x08020000) /* Base @ of Sector 5, 128 Kbytes */
#define ADDR_FLASH_SECTOR_6     ((uint32_t)0x08040000) /* Base @ of Sector 6, 128 Kbytes */
#define ADDR_FLASH_SECTOR_7     ((uint32_t)0x08060000) /* Base @ of Sector 7, 128 Kbytes */
#define ADDR_FLASH_SECTOR_8     ((uint32_t)0x08080000) /* Base @ of Sector 8, 128 Kbytes */
#define ADDR_FLASH_SECTOR_9     ((uint32_t)0x080A0000) /* Base @ of Sector 9, 128 Kbytes */
#define ADDR_FLASH_SECTOR_10    ((uint32_t)0x080C0000) /* Base @ of Sector 10, 128 Kbytes */
#define ADDR_FLASH_SECTOR_11    ((uint32_t)0x080E0000) /* Base @ of Sector 11, 128 Kbytes */

#define RAM_BUFFER_SIZE               ((uint32_t)30*1024)       /*KBytes*/

#define filename "LED.bin"

#define FLASH_USER_START_ADDR   ADDR_FLASH_SECTOR_4   /* Start @ of user Flash area */
#define FLASH_USER_END_ADDR     ADDR_FLASH_SECTOR_4  +  0x10000

uint32_t FLASH_Erase_Write(void);
static uint32_t GetSector(uint32_t Address);
void jumpToApp(void);
void UP_Data(void);

#endif 

BootLoader.c

#include "BootLoader.h"

extern ApplicationTypeDef Appli_state;
FRESULT res;
static FLASH_EraseInitTypeDef EraseInitStruct;

uint8_t RAM_Buffer[RAM_BUFFER_SIZE];
uint32_t APP_Size;
uint32_t FirstSector = 0, NbOfSectors = 0, Address = 0;
uint32_t SectorError = 0;
__IO uint32_t data32 = 0 , MemoryProgramStatus = 0;
uint8_t errorcode;
uint32_t *p;

uint8_t SystemUpdateFlag = 0, state = 0;
uint16_t t = 0;
		
typedef  void (*pFunction)(void);
pFunction Jump_To_Application;
uint32_t JumpAddress;

int fputc(int ch, FILE *f)		//用来打印信息的 便于观察
{
	HAL_UART_Transmit(&huart3, (uint8_t *)&ch, 1, 100);
	return ch;
}
uint32_t FLASH_Erase_Write(void)		//FLASH擦写
{
  uint32_t i = 0;
	
  HAL_FLASH_Unlock();
  FirstSector = GetSector(FLASH_USER_START_ADDR);
  NbOfSectors = GetSector(FLASH_USER_END_ADDR) - FirstSector + 1;
	
  EraseInitStruct.TypeErase = FLASH_TYPEERASE_SECTORS;
  EraseInitStruct.VoltageRange = FLASH_VOLTAGE_RANGE_3;
  EraseInitStruct.Sector = FirstSector;
  EraseInitStruct.NbSectors = NbOfSectors;
	if(HAL_FLASHEx_Erase(&EraseInitStruct, &SectorError) != HAL_OK)
  {
    errorcode = HAL_FLASH_GetError();		//擦除失败的扇区
	printf("errorcode %d", errorcode);
    Error_Handler();
  }
  __HAL_FLASH_DATA_CACHE_DISABLE();
  __HAL_FLASH_INSTRUCTION_CACHE_DISABLE();

  __HAL_FLASH_DATA_CACHE_RESET();
  __HAL_FLASH_INSTRUCTION_CACHE_RESET();

  __HAL_FLASH_INSTRUCTION_CACHE_ENABLE();
  __HAL_FLASH_DATA_CACHE_ENABLE();
	
	printf("HAL_OK\r\n");
	
  Address = FLASH_USER_START_ADDR;

  while (Address < FLASH_USER_END_ADDR)
  {
	p = (uint32_t *)&RAM_Buffer[i];
    if (HAL_FLASH_Program(FLASH_TYPEPROGRAM_WORD, Address, *p) == HAL_OK)	//FLASH写入
    {
      Address = Address + 4;
	  i = i + 4;
    }
    else
    {
	  printf("Address-error\r\n");
      Error_Handler();
    }
  }
  HAL_FLASH_Lock(); 

  Address = FLASH_USER_START_ADDR;		//校验
  MemoryProgramStatus = 0x0;
  printf("CHEAK\r\n");
  while (Address < FLASH_USER_END_ADDR)
  {
    data32 = *(__IO uint32_t*)Address;

    if (data32 != *(uint32_t*)RAM_Buffer)
    {
      MemoryProgramStatus++;
    }
    Address = Address + 4;
  }
	printf("CHEAK-finish\r\n");
	
	return HAL_OK;
}
static uint32_t GetSector(uint32_t Address)		//获取扇区
{
  uint32_t sector = 0;
  
  if((Address < ADDR_FLASH_SECTOR_1) && (Address >= ADDR_FLASH_SECTOR_0))
  {
    sector = FLASH_SECTOR_0;  
  }
  else if((Address < ADDR_FLASH_SECTOR_2) && (Address >= ADDR_FLASH_SECTOR_1))
  {
    sector = FLASH_SECTOR_1;  
  }
  else if((Address < ADDR_FLASH_SECTOR_3) && (Address >= ADDR_FLASH_SECTOR_2))
  {
    sector = FLASH_SECTOR_2;  
  }
  else if((Address < ADDR_FLASH_SECTOR_4) && (Address >= ADDR_FLASH_SECTOR_3))
  {
    sector = FLASH_SECTOR_3;  
  }
  else if((Address < ADDR_FLASH_SECTOR_5) && (Address >= ADDR_FLASH_SECTOR_4))
  {
    sector = FLASH_SECTOR_4;  
  }
  else if((Address < ADDR_FLASH_SECTOR_6) && (Address >= ADDR_FLASH_SECTOR_5))
  {
    sector = FLASH_SECTOR_5;  
  }
  else if((Address < ADDR_FLASH_SECTOR_7) && (Address >= ADDR_FLASH_SECTOR_6))
  {
    sector = FLASH_SECTOR_6;  
  }
  else if((Address < ADDR_FLASH_SECTOR_8) && (Address >= ADDR_FLASH_SECTOR_7))
  {
    sector = FLASH_SECTOR_7;  
  }
  else if((Address < ADDR_FLASH_SECTOR_9) && (Address >= ADDR_FLASH_SECTOR_8))
  {
    sector = FLASH_SECTOR_8;  
  }
  else if((Address < ADDR_FLASH_SECTOR_10) && (Address >= ADDR_FLASH_SECTOR_9))
  {
    sector = FLASH_SECTOR_9;  
  }
  else if((Address < ADDR_FLASH_SECTOR_11) && (Address >= ADDR_FLASH_SECTOR_10))
  {
    sector = FLASH_SECTOR_10;  
  }
  else /* (Address < FLASH_END_ADDR) && (Address >= ADDR_FLASH_SECTOR_11) */
  {
    sector = FLASH_SECTOR_11;
  }

  return sector;
}
void jumpToApp()
{
    if (((*(__IO uint32_t*)FLASH_USER_START_ADDR) & 0x2FFE0000 ) == 0x20000000)
    {
	  printf("ADDR == 0x20000000\r\n");
      JumpAddress = *(__IO uint32_t*) (FLASH_USER_START_ADDR + 4);
	  Jump_To_Application = (pFunction) JumpAddress;
	  __set_MSP(*(__IO uint32_t*) FLASH_USER_START_ADDR);
	  __HAL_UART_DISABLE(&huart3);				//关闭相应中断
	  __HAL_RCC_USB_OTG_FS_CLK_DISABLE();
	  __HAL_UART_CLEAR_FLAG(&huart3, UART_FLAG_TC);
	  __HAL_UART_DISABLE_IT(&huart3, UART_IT_RXNE);
	  Jump_To_Application();
    }
	printf("ADDR != 0x20000000\r\n");
}
void UP_Data(void)
{
	if(state == 0)
	{
		if(t < 2010)
		{
			if(SystemUpdateFlag == 0 && Appli_state == APPLICATION_READY)
			{
				printf("APPLICATION_READY\r\n");
				SystemUpdateFlag = 1;
				t = 2000;
				state = 1;
				res = f_mount(&USBHFatFS, (TCHAR const*)USBHPath, 0);
				if(res != FR_OK)
				{
					printf("U盘挂载失败  %d\r\n", res);
					Error_Handler();
				}
				else
				{
					printf("U盘挂载成功\r\n");
				}
				res = f_open(&USBHFile, filename, FA_READ);
				if(res != FR_OK)
				{
					printf("打开U盘文件失败  %d\r\n", res);
					Error_Handler();
				}
				else
				{
					printf("打开U盘文件成功\r\n");
				}
				res = f_read(&USBHFile, RAM_Buffer, sizeof(RAM_Buffer), (void *)&APP_Size);
				if(res != FR_OK)
				{
					printf("读取U盘文件失败  %d\r\n", res);
					Error_Handler();
				}
				else
				{
					printf("读取U盘文件成功\r\n");
				}	
				if((0<APP_Size) && (APP_Size<FLASH_USER_END_ADDR))
				{
					printf("FLASH_Erase\r\n");
					FLASH_Erase_Write();
					jumpToApp();
				}
				else
				{
					printf("APP_Size错误\r\n");
				}
				f_close(&USBHFile);
		//		FATFS_UnLinkDriver(USBHPath);		
			
			}
			else
			{
				printf("%d\r\n", t);		
				t++;
				HAL_Delay(1);
			}
			if(state == 0 && t > 2000)		//超过2s 读取U盘失败
			{
				state = 1;
				printf("DISCONNECT\r\n");
				jumpToApp();
			}

		}
		
	}
}	

四、APP:

1 APP程序只进行LED闪烁,只配置这三个功能就够了请添加图片描述
2 时钟树配置
请添加图片描述

3 工程配置
请添加图片描述
请添加图片描述
4 接下来是APP程序
(1)main.c 只放LED闪烁的功能请添加图片描述

(2)中断偏移地址 这个地址要和BootLoader程序里面flash的起始地址要一样请添加图片描述
(3)魔术棒配置请添加图片描述

请添加图片描述
请添加图片描述

D:\software\keil5\path \ARM\ARMCLANG\bin\fromelf.exe --bin --output ./APP/LED.bin ./APP/APP.axf

详细说明

D:\software\keil5\path \ARM\ARMCLANG\bin\fromelf.exe

formelf.exe是Keil自带的格式工具软件,其路径在你电脑的Keil安装盘的对应位置;

 

不要以为这里啥也没有,这里是有一个空格的;

--bin

输出bin文件的命令语句;

 

空格;

--output

输出文件的命令语句,也可以写成简写的形式:-o;

 

空格;

./APP/LED.bin

APP是我用于存放输出bin文件的文件夹(系统默认此路径从工程文件开始算起),LED.bin是我生成的bin文件;

 

空格;

./APP/APP.axf

APP是用于存放axf输入文件的文件夹,生成bin文件前,要先生成axf输入文件;


配置好后点击OK,编译一下工程,bin文件就生成了
请添加图片描述
添加全局变量 USER_VECT_TAB_ADDRESS

,USER_VECT_TAB_ADDRESS

变量前面有个逗号 英文的

请添加图片描述
将生成的bin文件放到U盘里请添加图片描述

五、实验现象:

BootLoader第一次下载是最好是进行一下全片擦除,之后使用扇区擦除请添加图片描述
(1)没有检测到U盘现象
蓝色是电源指示灯,红框才是实验灯(红色的)
请添加图片描述
串口打印信息
两秒没有检测到U盘打印的信息

没有检测到U盘


(2)检测到U盘现象
红色LED以100ms的时间在闪烁

升级成功现象

串口打印信息

检测到U盘

六、文件篇:

U盘升级STM32F407(文件篇)

  • 21
    点赞
  • 101
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 24
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 24
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

饼里个饼

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值