操作stm32 有使用官方库函数(参见stm32 开发环境MDK+库文件配置)和 直接操作寄存器的方法
直接操作寄存器的方法 会比库函数的方法效率更高 而且代码量会比较少 例如 在库函数下 配置一个GPIO口 需要
GPIO_InitTypeDef GPIO_InitStructure; //结构体 初始化
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_Init(GPIOA , &GPIO_InitStructure);
五行代码 而直接操作寄存器只需要:
RCC->APB2ENR|=1<<2; //使能PORTA时钟
GPIOA->CRL&=0XFFF0FFFF;
GPIOA->CRL|=0X00030000;//PA4 推挽输出
三行代码 而且实际上这三行代码可以配置8个GPIO口 可以看出直接操作寄存器也是比较方便的
使用直接操作寄存器的方法操作stm32 环境配置和库函数类似 相关MDK设置可以参考stm32 开发环境MDK+库文件配置 直接操作寄存器需要的文件结构 会少得多
STM32 直接操作寄存器 keil工程结构
Startup 包含的是stm32的 启动文件,与芯片Flash容量有关
Library 下有两个文件夹,src文件夹用于放置标准外设库驱动源文件(*.c)和 inc文件夹用于放置标准外设库驱动头文件(*.h)
User中包含的是项目的代码 和中断代码
Project 用于包含编译是时生成的一系列文件,Output 用来放置输出文件 .hex .axf,Listing用来放置Listing信息
需要说明的是 Startup里的启动文件需要根据不同的芯片选择不同的启动文件,这些启动文件在MDK的安装文件夹下可以找到 在MDK新建工程是选择了stm32的芯片型号后 MDK也会询问是否将启动文件添加到工程里
在MDk安装路径ARMStartupSTSTM32F10x的所有启动文件:
小容量产品是指闪存存储器容量在16K至32K字节之间的STM32F101xx、STM32F102xx和STM32F103xx微控制器。 选择 startup_stm32f10x_ld.s。
中容量产品是指闪存存储器容量在64K至128K字节之间的STM32F101xx、STM32F102xx和STM32F103xx微控制器。选择 startup_stm32f10x_md.s。
大容量产品是指闪存存储器容量在256K至512K字节之间的STM32F101xx和STM32F103xx微控制器。选择 startup_stm32f10x_hd.s。
容量大小可以通过芯片型号得知:
还需要说明的一点是在MDk安装路径ARMStartupST下有一个 STM32F10x.s的启动文件
STM32F10x.s 可以作为大部分stm32型号的芯片的启动文件,但是并不能适用所有的STM32型号。
STM32F10x.s是MDK提供的启动代码,从其里面的内容看来,里面定义了STM32的堆栈大小以及各种
中断的名字及入口函数名称,还有启动相关的汇编代码。它只定义了3个串口,4个定时器。
实际上STM32的系列产品有5个串口的型号,也只有有2个串口的型号,定时器也是,做多的有8个定时
器。
比如,如果你用的STM32F103ZET6,而启动文件用的是STM32F10x.s的话,你可以正常使用串口
1~3的中断,而串口4和5的中断,则无法正常使用。又比如,你TIM1~4的中断可以正常使用,而5~8
的,则无法使用。
和库函数操作类似 直接操作寄存器方法也需要先配置RCC时钟 配置中断等操作 这里提供一个配置函数,后面的例子中都会调用这个文件)
Library/src/system.c
#include#include"system.h"/************************************************************系统函数**功能:实现中断的初始化、RCC时钟初始化、Systick初始化以及延时函数等***********************************************************///设置向量表偏移地址//NVIC_VectTab:基址//Offset:偏移量voidNvic_SetVectorTable(u32NVIC_VectTab,u32Offset){//检查参数合法性assert_param(IS_NVIC_VECTTAB(NVIC_VectTab));assert_param(IS_NVIC_OFFSET(Offset));SCB->VTOR=NVIC_VectTab|(Offset&(u32)0x1FFFFF80);//设置NVIC的向量表偏移寄存器//用于标识向量表是在CODE区还是在RAM区}//设置NVIC分组//NVIC_Group:NVIC分组0~4总共5组voidNvic_PriorityGroupConfig(u8NVIC_Group){u32temp,temp1;//配置向量表#ifdefVECT_TAB_RAMNvic_SetVectorTable(NVIC_VectTab_RAM,0x0);#elseNvic_SetVectorTable(NVIC_VectTab_FLASH,0x0);#endiftemp1=(~NVIC_Group)&0x07;//取后三位temp1<<=8;temp=SCB->AIRCR;//读取先前的设置temp&=0X0000F8FF;//清空先前分组temp|=0X05FA0000;//写入钥匙temp|=temp1;SCB->AIRCR=temp;//设置分组}//设置NVIC//NVIC_PreemptionPriority:抢占优先级//NVIC_SubPriority:响应优先级//NVIC_Channel:中断编号//NVIC_Group:中断分组0~4//注意优先级不能超过设定的组的范围!否则会有意想不到的错误//组划分://组0:0位抢占优先级,4位响应优先级//组1:1位抢占优先级,3位响应优先级//组2:2位抢占优先级,2位响应优先级//组3:3位抢占优先级,1位响应优先级//组4:4位抢占优先级,0位响应优先级//NVIC_SubPriority和NVIC_PreemptionPriority的原则是,数值越小,越优先voidNvic_Init(u8NVIC_PreemptionPriority,u8NVIC_SubPriority,u8NVIC_Channel,u8NVIC_Group){u32temp;u8IPRADDR=NVIC_Channel/4;//每组只能存4个,得到组地址u8IPROFFSET=NVIC_Channel%4;//在组内的偏移IPROFFSET=IPROFFSET*8+4;//得到偏移的确切位置Nvic_PriorityGroupConfig(NVIC_Group);//设置分组temp=NVIC_PreemptionPriority<<(4-NVIC_Group);temp|=NVIC_SubPriority&(0x0f>>NVIC_Group);temp&=0xf;//取低四位if(NVIC_Channel<32)NVIC->ISER[0]|=1
curton 2019-10-10 22:31