GPIO位带操作
# 位带操作
# 什么是位带操作
关于什么是位带操作,其实包括野火或者正点原子的书上写的很清楚,就是为了实现对某个精确的比特的读写讲位带区的一个比特扩大为别名区的四个字节(32 bit)
不清楚的话可以参考:
- 《野火®零死角玩转STM32(库函数版)》14章
- 《正点原子:STM32F4标准库》5.2.1 节
在 SRAM 和 外设取均设有位带区。
关于换算公式可以了解一下,其实理解原理之后就能很好的理解到底是怎样从位带区的一个比特映射到别名区的。对了,一定不要忘了地址空间上寄存器的分布规律(就是四个字节分为一组,一组为一个寄存器,就在前面的文章)
# GPIO 位带操作
对 GPIO 进行位带操作可以方便的使用位操作控制 GPIO 的输入与输出
# GPIO 寄存器映射
要控制 GPIO 的输入与输出,需要控制 ODR 与 IDR 寄存器,所以首先对这些寄存器进行宏定义:
// GPIO ODR 和 IDR 寄存器地址映射
#define GPIOA_ODR_Addr (GPIOA_BASE+12) //0x4001080C
#define GPIOB_ODR_Addr (GPIOB_BASE+12) //0x40010C0C
#define GPIOC_ODR_Addr (GPIOC_BASE+12) //0x4001100C
#define GPIOD_ODR_Addr (GPIOD_BASE+12) //0x4001140C
#define GPIOE_ODR_Addr (GPIOE_BASE+12) //0x4001180C
#define GPIOF_ODR_Addr (GPIOF_BASE+12) //0x40011A0C
#define GPIOG_ODR_Addr (GPIOG_BASE+12) //0x40011E0C
#define GPIOA_IDR_Addr (GPIOA_BASE+8) //0x40010808
#define GPIOB_IDR_Addr (GPIOB_BASE+8) //0x40010C08
#define GPIOC_IDR_Addr (GPIOC_BASE+8) //0x40011008
#define GPIOD_IDR_Addr (GPIOD_BASE+8) //0x40011408
#define GPIOE_IDR_Addr (GPIOE_BASE+8) //0x40011808
#define GPIOF_IDR_Addr (GPIOF_BASE+8) //0x40011A08
#define GPIOG_IDR_Addr (GPIOG_BASE+8) //0x40011E08
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
下一步我们希望通过一个带参数的宏来方便的控制究竟是哪一个引脚在输入或输出
# GPIO 位操作
/* 计算公式 */
// 把“位带地址+位序号”转换成别名地址的宏
#define BITBAND(addr, bitnum) ((addr & 0xF0000000)+0x02000000+((addr & 0x00FFFFFF)<<5)+(bitnum<<2))
// 把一个地址转换成一个指针
#define MEM_ADDR(addr) *((volatile unsigned long *)(addr))
// 把位带别名区地址转换成指针
#define BIT_ADDR(addr, bitnum) MEM_ADDR(BITBAND(addr, bitnum))
// 单独操作 GPIO 的某一个IO 口,n(0,1,2...16),n 表示具体是哪一个IO 口
#define PAout(n) BIT_ADDR(GPIOA_ODR_Addr,n) //输出
#define PAin(n) BIT_ADDR(GPIOA_IDR_Addr,n) //输入
#define PBout(n) BIT_ADDR(GPIOB_ODR_Addr,n) //输出
#define PBin(n) BIT_ADDR(GPIOB_IDR_Addr,n) //输入
#define PCout(n) BIT_ADDR(GPIOC_ODR_Addr,n) //输出
#define PCin(n) BIT_ADDR(GPIOC_IDR_Addr,n) //输入
#define PDout(n) BIT_ADDR(GPIOD_ODR_Addr,n) //输出
#define PDin(n) BIT_ADDR(GPIOD_IDR_Addr,n) //输入
#define PEout(n) BIT_ADDR(GPIOE_ODR_Addr,n) //输出
#define PEin(n) BIT_ADDR(GPIOE_IDR_Addr,n) //输入
#define PFout(n) BIT_ADDR(GPIOF_ODR_Addr,n) //输出
#define PFin(n) BIT_ADDR(GPIOF_IDR_Addr,n) //输入
#define PGout(n) BIT_ADDR(GPIOG_ODR_Addr,n) //输出
#define PGin(n) BIT_ADDR(GPIOG_IDR_Addr,n) //输入
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
这么一来:PAout(5) = 1
就可以表示 GPIOA_Pin5
输出高电平
判断GPIOA_Pin5
是否正在输出高电平可以使用 if (PAin(5) == 1)
编辑 (opens new window)
上次更新: 2020/11/17, 11:11:00