MM32F0144的IAP实例

IAP的原理网络上有很多文章,我参考的是这一篇,写的比较好

这是最后一篇了,五精华无悬念:STM32的IAP方案 - stm32/stm8 - 电子工程世界-论坛 (eeworld.com.cn)

这里不再介绍原理了。本人看了原理,尝试了写了个简单的验证程序,结果跳转成功后中断不灵了,又查了很多网文,发现是uboot中是在中断中触发了跳转导致的异常,后来修改成直接在main函数中执行跳转,就成功了。这里记录一下关键点,并把工程源码也附上,供人参考。

UBOOT

#define APPLICATION_START_ADDR (uint32_t)(0x08001000)
typedef void (*pFunction)(void);

uint32_t JumpAddress = 0;
pFunction JumpToApplication;


s32 main(void)
{
	
		DELAY_Init();
        LED_Init();   //这里点灯的目的是方便观察运行在APP还是uboot上
		LED2_ON();
		DELAY_Ms(200);
		LED2_OFF();	
		DELAY_Ms(200);
	
		LED2_ON();
		DELAY_Ms(200);
		LED2_OFF();	
		DELAY_Ms(200);
	
		LED2_ON();
		DELAY_Ms(200);
		LED2_OFF();	
		DELAY_Ms(200);

//IAP跳转,不能在中断程序中执行,否则切换到APP后中断异常
		JumpAddress = *(volatile uint32_t *)(APPLICATION_START_ADDR + 4);
		JumpToApplication = (pFunction) JumpAddress;
		__set_MSP(*(volatile uint32_t *)APPLICATION_START_ADDR);
		JumpToApplication();
//IAP跳转

    while(1) 
		{			


    }

}

APP

#define APPLICATION_START_ADDR (uint32_t)(0x08001000)
s32 main(void)
{
		 

	memcpy((uint8_t *)SRAM_BASE, (uint8_t *)APPLICATION_START_ADDR, 48*4);	//这里复制中断向量表
	RCC_APB2PeriphClockCmd(RCC_APB2ENR_SYSCFG, ENABLE);				
	SYSCFG_MemoryRemapConfig(SYSCFG_MemoryRemap_SRAM); 
	
	
	DELAY_Init();
    LED_Init();
    KEY_Init();
    EXTI_Config();
    while(1) 
	{
			
			DELAY_Ms(200);
			LED1_TOGGLE();
			DELAY_Ms(200);


    }

}

这里0xC0是48预留给中断向量的

0x10000-0x1000=0xF000

0x2000-0xC0=0x1F40

源码下载0积分

https://download.csdn.net/download/hyhsandy1803/88876117

MM32F0144C6P是一款基于ARM Cortex-M0+内核的微控制器,它通常需要配合外部晶振来设定系统的时钟频率。如果你想要使用12MHz的晶振作为系统时钟源,首先你需要配置微控制器的晶振分频器(PLL)。以下是一个基本的代码示例,假设你正在使用Keil MDK这样的开发环境: ```c #include "stm32f0xx.h" // 包含STM32的头文件 void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct; RCC_ClkInitTypeDef RCC_ClkInitStruct; // 开启HSE (高性能内部振荡器) RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE; // HSE Oscillator RCC_OscInitStruct.HSEState = RCC_HSE_ON; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON; // 启用 PLL RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE; // 选择HSE作为PLL的输入源 RCC_OscInitStruct.PLL.PLLM = 8; // 分频因子M(大多数情况下为8或更高,取决于你的HSE频率) RCC_OscInitStruct.PLL.PLLN = 192; // 分频系数N,设置为192以得到12MHz的系统时钟 RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2; // 如果你的核心工作电压支持,可以考虑这个选项 RCC_OscInitStruct.PLL.PLLQ = 2; // Q值通常为2或4 if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) { while (1); // 错误处理,无限循环 } // 将时钟从PLL输出切换到系统时钟 RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2; RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK; // 设置sysclk为pll输出 RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; // AHB总线时钟除以1 RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4; // APB1时钟除以4 RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2; // APB2时钟除以2 if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_5) != HAL_OK) { while (1); // 错误处理,无限循环 } } int main(void) { // 其他初始化代码... SystemClock_Config(); // 初始化系统时钟 // 程序主体... } ``` 记得在实际项目中,要根据具体的硬件配置和芯片手册调整分频系数。执行`SystemClock_Config()`函数后,你的系统就会使用12MHz晶振设置了。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

hyhsandy1803

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

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

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

打赏作者

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

抵扣说明:

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

余额充值