具体配置过程: 第一步: 复位并配置向量表。 函数MYRCC_DeInit(); 下面对该函数进行分析: (1) 设置外设复位寄存器:RCC->APB1RSTR = 0x00000000 该寄存器中包含dac,电源复位,定时器等外设复位设置,某位为1表示对相应外设复位。开机启动时将该寄存器数据清空。 (2) 设置外设复位寄存器:RCC->APB2RSTR = 0x00000000 同第一步外设复位寄存器的设置。 解答: RCC->APB1RSTR = 0x00000000;//复位结束
RCC->APB2RSTR = 0x00000000;
这里的“复位结束”具体是什么意思??我把它注释掉后发现也是可以运行的 1是复位.0当然是不复位了
不复位那就是复位结束了. (3) 睡眠模式闪存和sram时钟使能,其他关闭。用于使用sram。 Sram相当于pc的内存。 STm32有三种启动模式: 1,ISP模式.这种模式就是STM32复位后就执行固化在内部的BOOTLOADER程序(固化的,我们无法读写.),然后等待串口数据,从而实现串口bootloader功能.
这种模式不会从用户存储区启动(除非用串口控制其从0X08000000启动),所以在更新了代码之后,需要设置为其他模式(FLASH模式).
2,FLASH启动模式.这种模式直接从0X08000000启动,也就是我们自己编写的代码的启动方式了.正常情况都应该用这种.
3,SRAM启动模式.这种模式我没有用过,是从0X20000000启动的,也就是说在sram模式开始之前,你要确保SRAM里面已经有代码了,否则就是死机. RCC->AHBENR = 0x00000014 (4) 设置外设时钟使能寄存器: RCC->APB1ENR = 0x00000000; RCC->APB2ENR = 0x00000000; 将所有外设全部关闭 (5) 使能内部高速HSION。 RCC->CR |=0x00000001; stm32的时钟启动过程。
启动过程是:
1,首先使用内部时钟(这也是为什么你不接晶振也可以下载代码了)。
2,尝试开启外部时钟.
3,如果开启成功,则使用外部时钟,否则使用内部。
4,做其他事情。
当然以上代码都需要你自己写代码实现,当然内部时钟是默认的时钟,你不开启也可以. (6) 复位SW,HPRE,PPRE1,PPRE2,ADCPRE,MCO RCC->CFGR &= 0xF8FF0000; 这步有什么意思呢,我的理解是。Cfgr寄存器主要用于对时钟分频的控制,见下图:
通过该步的配置: 首先配置MCO无输出,MCO是什么呢?是指可以将stm32的内部时钟通过IO口引脚输出出去,如上图就可以看到,对cfgr的配置,可以有四种mco输出,分别是将pllclk两分频后输出,hsi(片内时钟)输出等。 其次:配置ADCPRE就是上图中AHB分频器线面的ADC 再次:配置ppre2也就是高速外部时钟APB2,这里设成不分频。高速外部时钟主要驱动一些高速外设,这个在APB2ENR时钟控制寄存器中有介绍 再次:配置PPRE1配置低速外部时钟分频APB1这里也全部设成不分频。 再次:配置HPRE。这几个位主要用来配置AHB这个寄存器的分频系数这里也设置成不分频。也就是说上图SYSCLK经AHB没有分频。 最后:配置SW,以及SWS。表示启用HIS作为系统时钟。 到这一步,经过分析得知,RCC->CFGR &= 0xF8FF0000;主要是用来配置ahb等各个分频器的设置,以及将片内时钟作为系统内部时钟。 (6) 关闭HSEON,CSSON,PLLON RCC->CR &= 0xFEF6FFFF; 通过分析CR寄存器可以看出,该寄存器主要涉及三个时钟PLL,CSS,HSE。 (7) 复位HSEBYP. RCC->CR &= 0xFFFBFFFF;这一步有什么作用呢?查询数据手册57页可知,外部时钟源HSE有两种模式,HSEBYP设置为0时,是选择外部晶体作为外部时钟源这种时钟更加精准,当然也是和外部电路有关的。当然因为第(6)步已经设置了HSEON关闭了,所以这一步才可自由设置HSEBYP。 (8) 复位PLLSRC,PLLXTPRE,PLLMUL and USBPRE RCC->CFGR &= 0xFF80FFFF; 注意:在这一部中可能会有这样的疑问: RCC->CFGR &= 0xFF80FFFF;
PLLSRC=0 HSI振荡器时钟经2分频后作为PLL输入时钟
PLLXTPRE=0,HSE分频器作为PLL输入,HSE不分频
这样不冲突吗? 答案是:以最后配置为准,就是最后一次配置会改变前一次的配置,所以说以最后一次配置为准。 也就是说后文还有其他代码对其进行定义。那干嘛还要怎么重复配置呢? 有时候是有用的。比如你想让stm32超频一会,然后又恢复正常运行,这就有用了。 (9) 关闭所有中断 RCC->CIR = 0x00000000; (10) 配置向量表 #ifndef VECT_TAB_RAM MY_NVIC_SetVectorTable(NVIC_VectTab_RAM,0x0); #else MY_NVIC_SetVectorTable(NVIC_VextTab_FLASH,0x0); #endif 下面对该函数分析: //函数功能:设置向量表偏移地址 //NVIC_VectTab:基址 //Offset:偏移量 void MY_NVIC_SetVectorTable(u32 NVIC_VectTab, u32 Offset)
{
//检查参数合法性
assert_param(IS_NVIC_VECTTAB(NVIC_VectTab));
assert_param(IS_NVIC_OFFSET(Offset));
SCB->VTOR = NVIC_VectTab|(Offset & (u32)0x1FFFFF80);//设置NVIC的向量表偏移寄存器
//用于标识向量表是在CODE区还是在RAM区
}
前面两行是用来检查参数合法性,这里不作分析。重点看第三行 配置这个向量表有什么用?相见cortexm3权威指南113页向量表的解释 这里 #define NVIC_VectTab_RAM ((u32)0x20000000) #define NVIC_VectTab_FLASH ((u32)0x08000000) Offset的值为0x0,为偏移地址,地址必须能被64 * 4 = 256整除,具体请看权威手册113页 SCB->VTOR = NVIC_VectTab|(Offset & (u32)0x1FFFFF80);//设置NVIC的向量表偏移寄存器的疑问如下: SCB->VTOR = NVIC_VectTab|(Offset & (u32)0x1FFFFF80);//设置NVIC的向量表偏移寄存器。
既然是设置NVIC的向量表偏移量,为什么还要和NVIC_VectTab相或呢。只设置OFFSET不就可以了吗,另外VTOR设置只有BIT【28:7】有作用啊,相或以后也放不下这么多位吧? 这个是基址。
那个7~28的,你能定义一个28位的数据出来嘛? VTOR设置只有BIT【28:7】,你把(u32)0x1FFFFF80二进制看看是不是【28:7】。
然后再看下面一段话:
在<<权威指南>>第一百零四页,有这么一段话:
NVIC 中有一个寄存器,称为“向量表偏移量寄存器”(在地址0xE000_ED08处),通过修改它的值就能定位向量表。但必须注意的是:向量表的起始地址是有要求的:必须先求出系统中共有多少个向量,再把这个数字向上增大到是2的整次幂,而起始地址必须对齐到后者的边界上。例如,如果一共有32个中断,则共有32+16(系统异常)=48个向量,向上增大到2的整次幂后值为64,因此地址
地址必须能被64*4=256整除,从而合法的起始地址可以是:0x0, 0x100, 0x200等。
向量表偏移量寄存器,也就是SCB->VTOR.它的第29位,用来标识向量表是在CODE区还是RAM区,从而0X1,就是最高3位不去动,这好理解. 但是低位,根据上面这段话的理解,STM32自己有60个中断,加上CM3的16个,总共有76个中断,扩大到2的整次幂,那就是128,然后再乘以4,得到512,也就是0X200.根据这样计算,合法的偏移地址应该是0X0,0X200,0X400,0X600...因此,在此处应该&0X1FFF FE00.才对.
以上是我的理解.实际上确是&0X1FFF FF80;这点,我也有疑问. 答案:cortex-m3权威指南上介绍 bit 28-7为向量表的起始地址。所以低7位没有用到,所以&0X80,为的就是将低七位清零。但这里写&0X1FFF FE00,也能达到清零的目的。至于地址必须是512的整数只要offset这个参数注意就可以了。
|