零基础自学STM32-复习篇2——使用结构体封装GPIO寄存器

我们首先要了解寄存器的一个特点,他不是只针对一个外设,而是所有的外设都。
就拿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)
  • 1
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值