求助:#define GPIOA ((GPIO_TypeDef *) GPIOA_BASE)

2019-10-12 14:07发布

原子哥资料上说    #define     GPIOA         ((GPIO_TypeDef *) GPIOA_BASE)    这句语句的意思是GPIOA指向
地址GPIOA_BASE(即0x40010800)。请问定义这条语句后GPIOA怎么就指向GPIOA_BASE(即0x40010800)了,
哪位大神能回答下啊?
或者解释下  ((GPIO_TypeDef *) GPIOA_BASE) 这个强制类型转换也可以啊。
谢谢!
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
12条回答
it_do_just
1楼-- · 2019-10-12 15:41
((GPIO_TypeDef *) GPIOA_BASE)   //这里将GPIOA基地址强制转为了GPIO_TypeDef这个类型的指针

//看看GPIO_TypeDef这个结构体,成员按照定义时的顺序依次存储在连续的内存空间。

typedef struct

{

  __IO uint32_t CRL;      //偏移量0x00

  __IO uint32_t CRH;      //偏移量0x04

  __IO uint32_t IDR;       //偏移量0x08

  __IO uint32_t ODR;      //偏移量0x0C

  __IO uint32_t BSRR;     //偏移量0x10

  __IO uint32_t BRR;      //偏移量0x14

  __IO uint32_t LCKR;     //偏移量0x18

} GPIO_TypeDef;

这样每操作一个结构体成员就达到了操作对应寄存器的地址跟寄存器可以对应上
it_do_just
2楼-- · 2019-10-12 16:16
你可以看看STM32中文参考手册,里面有讲到GPIO的寄存器有哪些,是不是跟上面的CRL,CRH,IDR....这些寄存器对应上了,包括偏移地址,根据手册查询到GPIOA的基地址后,强制转为这个结构类型,根据偏移地址可以对应到各个指定的寄存器,那么我们操作这个结构体就达到了操作GPIO的寄存器,关键还是C语言,建议再去看看
最珍贵年轻的心
3楼-- · 2019-10-12 18:29
你把GPIO_Type_Def  看作int类型  那么意思就是说GPIOA就是等于 (int*)BASE,将它强制性转化成了int型地址,C语言指针要好好看看。我这么说你是否理解了呢
shibinjie
4楼-- · 2019-10-12 20:34
 精彩回答 2  元偷偷看……
shibinjie
5楼-- · 2019-10-12 21:40
回复【4楼】最珍贵年轻的心:
---------------------------------
谢谢回答,。(int*)BASE,将它强制性转化成了int型地址,对这句不是很理解啊。(int*)BASE应该是一个指向int类型的指针吧?
it_do_just
6楼-- · 2019-10-12 23:29
回复【5楼】shibinjie:
---------------------------------
#define ERIPH_BASE           ((uint32_t)0x40000000)

#define APB2PERIPH_BASE       (PERIPH_BASE + 0x10000)

GPIOA_BASE = (APB2PERIPH_BASE + 0x0800) = ERIPH_BASE + 0x10000 + 0x0800 = 0x40000000 + 0x10000 + 0x0800

                      = 0x40010800

0x40010800代表一个地址值,那请问你怎么去操作他?直接给这个赋1他就让这个IO口输出1吗?显示不行,因为这个只是GPIOA的基地址,他还有很多个寄存器,每个寄存器有自己的作用,那那些寄存器的地址是多少呢?每个寄存器地址都有个偏移量,基地址+偏移量=寄存器地址,那么我们给寄存器赋值就可以达到控制GPIOA的目的,至于为什么强制转换为(GPIO_TypeDef *)类型,因为GPIO_TypeDef定义了几个变量,每个变量代表一个寄存器,你把这个结构体的第一个元素当成0x40010800,那么再往后第二个元素不就是0x40010800+4(因为u32占4字节),就是下一个寄存器的地址,这样可以达到操作结构体就操作寄存器的目的,建议你再好好学下C吧

一周热门 更多>