使用寄存器映射的方式点亮LED灯

使用寄存器映射的方式点亮LED灯

  1. 第一步,新建工程文件,在《STM32使用寄存器直接地址点亮LED》那篇文章中已经详细的写过了,所以在这里就不在重复的编写了。
  2. 思路和《STM32使用寄存器直接地址点亮LED》是一样的,都是分为三步:
    - [1] 开启时钟。
    - [2] 设置端口的输入输出模式。
    - [3] 设置端口的输出数值。
    但是在《STM32使用寄存器直接地址点亮LED》这篇文章中,当我们需要操作寄存器的时候需要去帮助文件中查找寄存器的地址,并且写地址也不助于代码的可读性,所以我们需要采用寄存器映射的方式来记录下这些寄存器的地址。接下来就是进行寄存器的映射代码的编写
  3. 寄存器映射代码的编写
    从下表可以得到各个寄存器的地址,这样我们进行一个宏定义,把这些地址都以好理解的英文来表示,在后续的使用中我们便可以直接调用英文字母来实现。
    寄存器映射表
    在这里插入图片描述
    编写的宏定义如下:
//这里定义的时候不需要定义为指针的类型
#define DEVICE_BASE_ADRESS (unsigned int )0x40000000
#define APB1_BASE_ADRESS DEVICE_BASE_ADRESS
#define APB2_BASE_ADRESS (APB1_BASE_ADRESS+0x10000)
#define AHB_BASE_ADRESS (APB2_BASE_ADRESS+0x20000)

#define GPIOB_BASE_ADRESS (APB2_BASE_ADRESS+0x0C00)
#define RCC_BASE_ADRESS (AHB_BASE_ADRESS+0x1000)

查找STM32的帮助文档,我们可以找到寄存器相对与GPIOB的偏移地址
寄存器的偏移地址
从而可以得到寄存器的地址,然后在进行宏定义。

//定义寄存器
#define RCC_APB2ENR *(unsigned int *)(RCC_BASE+0x18)
#define RCC_CR *(unsigned int *)(RCC_BASE_ADRESS+0x00)
#define RCC_CFGR *(unsigned int *)(RCC_BASE_ADRESS+0x04)
#define RCC_CIR *(unsigned int *)(RCC_BASE_ADRESS+0x08)
#define RCC_APB2RSTR *(unsigned int *)(RCC_BASE_ADRESS+0x0C)
#define RCC_APB1RSTR *(unsigned int *)(RCC_BASE_ADRESS+0x10)
#define RCC_AHBENR *(unsigned int *)(RCC_BASE_ADRESS+0x14)
#define RCC_APB2ENR *(unsigned int *)(RCC_BASE_ADRESS+0x18)
#define RCC_APB1ENR *(unsigned int *)(RCC_BASE_ADRESS+0x1C)
#define RCC_BDCR *(unsigned int *)(RCC_BASE_ADRESS+0x20)
#define RCC_CSR *(unsigned int *)(RCC_BASE_ADRESS+0x24)

#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)

这样就可以用寄存器的代码来代替《STM32使用寄存器直接地址来点亮LED》文章中的地址符号。

  //开启GPIOB的时钟RCC_AHB2ENR
  RCC_APB2ENR|= (1<<3);
  //设置GPIOB的端口类型GPIOB_CRL
  GPIOB_CRL |=(1<<0);
  //设置GPIOB的端口的输出\输入值GPIOB_BSRR和GPIOB_BRR
  //灯亮
  GPIOB_BRR |=(1<<0);
  //灯灭
  //GPIOB_BSRR |=(1<<0);

但是这只是针对一个端口来说,当对多个端口进行操作时需要定义很多这样的宏定义,所以我们可以继续进行精简,我们可以定义一个GPIO_Type类型的结构体和RCC_TypeDef的结构体,用来存放这些GPIO的寄存器和RCC的寄存器。

typedef unsigned int uint32_t;
typedef unsigned int uint16_t;
typedef struct
{
	uint32_t  CRL;
	uint32_t  CRH;
	uint32_t  IDR;
	uint32_t  ODR;
	uint32_t  BSRR;
	uint32_t  BRR;
	uint32_t  LCKR;
}GPIO_TypeDef;


typedef struct
{
	uint32_t  CR;
	uint32_t  CFGR;
	uint32_t  CIR;
	uint32_t  APB2RSTR;
	uint32_t  APB1RSTR;
	uint32_t  AHBENR;
	uint32_t  APB2ENR;
	uint32_t  APB1ENR;
	uint32_t  BDCR;
	uint32_t  CSR;
}RCC_TypeDef;

然后在把GPIOB_BASE_ADRESS转化为GPIO_TypeDef类型的变量把RCC_BASE_ADRESS 转化为RCC_TypeDef类型的变量,作为结构体的首地址。

#define GPIOB ((GPIO_TypeDef *)GPIOB_BASE)
#define RCC ((RCC_TypeDef *)RCC_BASE)

这样就可以通过调用结构体的方式来进行相应的操作。

      //配置GPIOB端口的时钟
	  RCC->APB2ENR |= (1<<3);
	  //配置IO口为推挽输出模式,速度为10MHz
	  GPIOB->CRL &=~((0x0f)<<0);
	  GPIOB->CRL |=(1<<0);
	  //配置PB0口输出0
	  GPIOB->ODR &=~(1<<0);
	  GPIOB->ODR |=(1<<0);
  1. 下载程序,灯被点亮。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
嵌入式系统中,GPIO(General Purpose Input Output)是通用输入输出引脚,可以用于控制外部设备的状态。LED是常见的外部设备之一,可以通过GPIO点亮。 下面是点亮LED的详细步骤: 1. 确定LED的连接方式,即将LED的正极连接到GPIO引脚,将负极连接到地。有些LED还需要加上一个限流电阻,以避免电流过大损坏LED。 2. 在嵌入式系统中,通过GPIO控制寄存器(GPIO control register)设置GPIO引脚的工作模式。一般来说,需要将GPIO引脚设置为输出模式(output mode)。 3. 将GPIO引脚的输出状态设置为高电平(high level),即将GPIO寄存器中相应的位设置为1,这样LED就会点亮。 下面是一个简单的C语言程序,用于点亮GPIO引脚连接的LED: ```c #include <stdio.h> #include <stdlib.h> #include <fcntl.h> #include <unistd.h> #include <sys/mman.h> #define GPIO_BASE 0x3F200000 // GPIO控制器的物理地址 #define GPIO_SIZE 0x1000 // GPIO控制器的大小 #define GPIO_PIN 17 // GPIO引脚号 int main() { // 打开/dev/mem文件获取内存映射地址 int fd = open("/dev/mem", O_RDWR | O_SYNC); if (fd < 0) { perror("open"); exit(1); } // 映射GPIO控制器的物理地址 void *gpio_map = mmap( NULL, GPIO_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, GPIO_BASE ); if (gpio_map == MAP_FAILED) { perror("mmap"); exit(1); } // 根据GPIO引脚号计算GPIO寄存器的偏移地址 unsigned int *gpio = (unsigned int *)gpio_map; int gpio_offset = GPIO_PIN / 32; int gpio_bit = GPIO_PIN % 32; unsigned int gpio_mask = 1 << gpio_bit; // 将GPIO引脚设置为输出模式 gpio[gpio_offset] |= (1 << (gpio_bit + 1)); // 将GPIO引脚的输出状态设置为高电平 gpio[gpio_offset + 7] |= gpio_mask; // 休眠1秒钟 sleep(1); // 将GPIO引脚的输出状态设置为低电平 gpio[gpio_offset + 10] |= gpio_mask; // 取消内存映射 munmap(gpio_map, GPIO_SIZE); // 关闭/dev/mem文件 close(fd); return 0; } ``` 这个程序使用了内存映射(memory mapping)技术,将GPIO控制器的物理地址映射到用户空间的虚拟地址,方便对GPIO寄存器进行读写操作。程序中使用了sleep函数来让LED保持点亮状态1秒钟,然后再将LED熄灭。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值