赞
踩
我们首先要了解寄存器的一个特点,他不是只针对一个外设,而是所有的外设都。
就拿GPIO的CRL,ODR寄存器来说
对于GPIOA——GPIOE都有一组功能相同的寄存器只是地址不一样而已A(主要是外设地址不同,比如GPIOA的外设地址和GPIOB的外设地址就不一样,即便寄存器的偏移量一样的),没必要每个寄存器都配置一遍。
这里就引入一个基本操作那就是:使用结构体对GPIO寄存器进行一次封装
主要参考:野火的《开发指南》
typedef unsigned int uint32_t; typedef unsigned short int uint16_t; // unsigned int 占32位unsigned short int占16位 //下面使用结构体来封装 /*********************************/ //定义一个结构体变量使用关键字struct,并将这个结构体命名为GPIO_Typdef如下(不懂结构体的可以去找个视频看看看,不必深究会用就行。) typedef struct { uint32_t CRL;//GPIO端口配置低寄存器器 uint32_t CRH;//GPIO端口配置高寄存器 uint32_t IDR;//GPIO端口输入寄存器 uint32_t ODR;//GPIO端口输出寄存器 uint32_t BSRR;//GPIO端口位置位/清除寄存器 uint32_t BRR;//GPIO端口位清除寄存器 uint32_t LCKR;//GPIO端口配置锁定寄存器 }GPIO_TypeDef;
这里还使用了typedef关键字命名了所创建的结构体类型为GPIO_TypeDef
其中结构体成员有7个变量,变量名正是所对应的寄存器名字。
C语言规定了结构体变量的存储空间是连续的(这个正好和我们的寄存器的地址是连续的特点相对应起来。这一点很重要,我们在封装的时候一定要按顺序去封装)。

比如我们定义的这个结构体GPIO_TypeDef。这个结构体的首个寄存器的地址就是CRL寄存器的地址也就是0x4001 0C00(这个地址也是GPIOB这个外设的总线地址),那么结构体第二个成员CRH的地址就是0x4001 0C00再加上一个0x04这个偏移量,为什么?
我们的寄存器是32位的,也就是4个字节为一个寄存器的存储空间,从CRL到CRH加上0x04也就是CRH的地址。其他成员也是相应的去加上偏移量就可以。(注意这个偏移量是相对于总线APB2的基地址的偏移量)
我们这里依然以代码为分析对象
//这里是stm32f103.h头文件 //用来存放stm32寄存器映射的代码 //GPIOB //外设基地址 peripheral # define PERIPHBASE ((unsigned int)0x40000000) //片内外设基地址 //总线基地址 # define APB1PERIPH_BASE PERIPHBASE //APB1总线基地址与片内外设基地址是一样的 # define APB2PERIPH_BASE (PERIPHBASE + 0x10000) //APB2总线基地址 # define AHBPERIPH_BASE (PERIPHBASE + 0x20000) //AHB总线基地址(为了后面操作RCC时钟),这里采用的DMA1的地址作为基地址 # define RCC_BASE (AHBPERIPH_BASE + 0x1000) //RCC复位时钟控制地址,想让IO工作必须配置 # define GPIOB_BASE (APB2PERIPH_BASE + 0x0C00) //GPIOB地址,操作GPIO的前提是找到总线地址 # define RCC_APB2ENR *(unsigned int* )(RCC_BASE + 0x18) //APB2外设时钟使能,想让IO工作必须配置 0x18就是APB2使能寄存器相对于RCC时钟总线的偏移量 //下面的代码使用宏定义重新命名一个指针变量,并通过操作寄存器的绝对地址指针从而操作寄存器工作 //# define GPIOB_CRL *(unsigned int*)(GPIOB_BASE + 0x00) //端口配置地寄存器 //# define GPIOB_CRH *(unsigned int*)(GPIOB_BASE + 0x04) //端口配置高寄存器 //# define GPIOB_ODR *(unsigned int*)(GPIOB_BASE + 0x0C) //数据输出寄存器 //# define GPIOB_IDR *(unsigned int*)(GPIOB_BASE + 0x08) //输入数据寄存器 //# define GPIOB_BSRR *(unsigned int*)(GPIOB_BASE + 0x10) //端口位设置/清除寄存器 //# define GPIOB_BRR *(unsigned int*)(GPIOB_BASE + 0x14) //端口位清除寄存器 //# define GPIOB_LCKR *(unsigned int*)(GPIOB_BASE + 0x18) //端口配置锁定寄存器 typedef unsigned int uint32_t;//声明一个32位的变量类型 typedef unsigned short uint16_t;//声明一个16位的变量类型 typedef struct { /* 这里解释一下为什么可以直接操作而不需要加偏移量 我们的stm32是32位单片机,每次操作32位数,而寄存器也是每个寄存器占用4个字节,也正好是32位, 创建结构体本身就会分配一个连续的内存空间,而我们定义了每个成员作为32位的变量,那正好跟寄存器本身的地址相对应起来。 */ uint32_t CRL; uint32_t CRH; uint32_t IDR; uint32_t ODR; uint32_t BSRR; uint32_t BRR; uint32_t LCKR; }GPIO_TypeDef; //让GPIOB的地址GPIOB_BASE转化为一个结构体类类型的变量,并重新命名为GPIOB # define GPIOB ((GPIO_TypeDef*)(GPIOB_BASE))
//32部分 #include "stm32f10x.h" void SystemInit(void);//不用管这条代码 int main (void) { # if 0 //使用这个语句相当于注释掉下面的代码 //使用寄存器绝对地址操作 *(unsigned int *)0x40021018 |=((1)<<3); //打开gpiob端口时钟 //*(unsigned int *)0x40010C0C &=~(1<<0); //Data_Output CRL_REG 通用推挽输出模式 *(unsigned int *)0x40010C0C &=~((1)<<0); //Data_Output CRL_REG *(unsigned int *)0x40010C00 |=((1)<<(4*0)); //Set_Output ODR_REG //数据输出寄存器 #elif 0 //注释掉下面的代码 RCC_APB2ENR |= ((1)<<3); //GPIOB时钟使能,在头文件中用宏定义声明过了 GPIOB_CRL |= ((1)<<(4*0)); //端口配置低寄存器 GPIOB_ODR &= ~(unsigned int)(1<<0); //数据输出寄存器 //GPIOB_ODR |= (1<<0); #elif 1 //执行这部分代码,使用结构封装好的函数去配置寄存器的值(重点掌握) RCC_APB2ENR |= ((1)<<3); //GPIOB时钟使能 GPIOB->CRL |= ((1)<<(4*0)
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。