编写bootloader程序:
- 直接复制下面代码到自己程序中。
typedef void (*iapfun)(void); //定义一个函数类型的参数.
iapfun jump2app;
//设置栈顶地址
//addr:栈顶地址
__asm void MSR_MSP(u32 addr)
{
MSR MSP, r0 //set Main Stack value
BX r14
}
//跳转到应用程序段
//appxaddr:用户代码起始地址.
void iap_load_app(u32 appxaddr)
{
delay_ms(500);
if(((*(vu32*)appxaddr)&0x2FF00000)==0x20000000) //检查栈顶地址是否合法.
{
debug("跳转到主程序\r\n");
GPIO_RESET(USB_EN);
delay_ms(500);
jump2app=(iapfun)*(vu32*)(appxaddr+4); //用户代码区第二个字为程序开始地址(复位地址)
/* 关闭全局中断*/
__set_PRIMASK(1);
/* 关闭滴答定时器,复位到默认值*/
SysTick->CTRL = 0;
SysTick->LOAD = 0;
SysTick->VAL = 0;
/* 设置所有时钟到默认状态 */
HAL_RCC_DeInit();
/* 关闭所有中断,清除所有中断挂起标志*/
for(u8 i=0;i<8;i++)
{
NVIC->ICER[i] = 0xFFFFFFFF;
NVIC->ICPR[i] = 0xFFFFFFFF;
}
/*使能全局中断 */
__set_PRIMASK(0);
/* 在RTOS工程,这条很重要,设置为特权模式,使用MSP指针*/
__set_CONTROL(0);
MSR_MSP(*(vu32*)appxaddr); //初始化APP堆栈指针(用户代码区的第一个字用于存放栈顶地址)
jump2app(); //跳转到APP.
}else
{
debug("栈顶地址不合法\r\n");
}
}
2 . 编写程序,接收应用程序的BIN文件,通过FLASH WRITE函数写入到目标扇区
3. 跳转到应用程序
iap_load_app((u32)0x08005000); // 跳转到主程序
计算应用程序的大小:
- 方法一:
查看编译输出计算:
- 方法二:
输出文件中找到.map文件,文件最后有计算结果
应用程序
- 设置程序存放的起始地址:
需要计算APP程序大小和bootloader程序大小,根据芯片扇区,合理安排位置。
如果0x80000000存放bootloader,需要计算出bootloader的大小,然后在后面地址存放应用程序。
我的bootloader大小是19k多,留出20K的空间。在0x8005000的位置存放该应用程序。 - 修改KEIL设置如下,填写起始保存地址,计算出空间大小。
3.
在main函数开始位置添加:
SCB->VTOR = ((u32)0x08005000);
4. 设置KEIL中自带的工具,编译工程,生成.bin文件
设置环境变量:
KEIL软件输入命令
fromelf.exe --bin -o ..\MDK-ARM\$PROJECT$\$PROJECT$.bin ..\MDK-ARM\$PROJECT$\$PROJECT$.axf
编译后生成下载所需的bin文件。