OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 11251|回复: 13

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

[复制链接]

38

主题

196

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1174
金钱
1174
注册时间
2015-11-9
在线时间
248 小时
发表于 2015-11-16 12:34:43 | 显示全部楼层 |阅读模式
5金钱
原子哥资料上说    #define     GPIOA         ((GPIO_TypeDef *) GPIOA_BASE)    这句语句的意思是GPIOA指向
地址GPIOA_BASE(即0x40010800)。请问定义这条语句后GPIOA怎么就指向GPIOA_BASE(即0x40010800)了,
哪位大神能回答下啊?
或者解释下  ((GPIO_TypeDef *) GPIOA_BASE) 这个强制类型转换也可以啊。
谢谢!

最佳答案

查看完整内容[请看2#楼]

((GPIO_TypeDef *) GPIOA_BASE)   //这里将GPIOA基地址强制转为了GPIO_TypeDef这个类型的指针 //看看GPIO_TypeDef这个结构体,成员按照定义时的顺序依次存储在连续的内存空间。 typedef struct {   __IO uint32_t CRL;      //偏移量0x00   __IO uint32_t CRH;      //偏移量0x04 &nbs ...
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

15

主题

786

帖子

5

精华

论坛大神

Rank: 7Rank: 7Rank: 7

积分
3223
金钱
3223
注册时间
2015-7-26
在线时间
811 小时
发表于 2015-11-16 12:34:44 | 显示全部楼层
((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;

这样每操作一个结构体成员就达到了操作对应寄存器的地址跟寄存器可以对应上
我的博客:http://blog.csdn.net/itdo_just
回复

使用道具 举报

15

主题

786

帖子

5

精华

论坛大神

Rank: 7Rank: 7Rank: 7

积分
3223
金钱
3223
注册时间
2015-7-26
在线时间
811 小时
发表于 2015-11-16 12:56:47 | 显示全部楼层
你可以看看STM32中文参考手册,里面有讲到GPIO的寄存器有哪些,是不是跟上面的CRL,CRH,IDR....这些寄存器对应上了,包括偏移地址,根据手册查询到GPIOA的基地址后,强制转为这个结构类型,根据偏移地址可以对应到各个指定的寄存器,那么我们操作这个结构体就达到了操作GPIO的寄存器,关键还是C语言,建议再去看看
我的博客:http://blog.csdn.net/itdo_just
回复

使用道具 举报

4

主题

19

帖子

0

精华

初级会员

Rank: 2

积分
68
金钱
68
注册时间
2015-10-12
在线时间
3 小时
发表于 2015-11-16 13:26:52 | 显示全部楼层
你把GPIO_Type_Def  看作int类型  那么意思就是说GPIOA就是等于 (int*)BASE,将它强制性转化成了int型地址,C语言指针要好好看看。我这么说你是否理解了呢
回复

使用道具 举报

38

主题

196

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1174
金钱
1174
注册时间
2015-11-9
在线时间
248 小时
 楼主| 发表于 2015-11-16 15:25:25 | 显示全部楼层
回复【3楼】229382777@qq.com:
---------------------------------
谢谢。根据基地址操作连续寄存器我都能理解。只是对这个((GPIO_TypeDef *) GPIOA_BASE)强制转换不是很理解。((GPIO_TypeDef *) GPIOA_BASE)是不是就是((GPIO_TypeDef *) 0x40010800),对0x40010800这个数强制转换成指向GPIO_TypeDef结构体类型的指针是什么意思?
回复

使用道具 举报

38

主题

196

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1174
金钱
1174
注册时间
2015-11-9
在线时间
248 小时
 楼主| 发表于 2015-11-16 15:27:25 | 显示全部楼层
回复【4楼】最珍贵年轻的心:
---------------------------------
谢谢回答,。(int*)BASE,将它强制性转化成了int型地址,对这句不是很理解啊。(int*)BASE应该是一个指向int类型的指针吧?
回复

使用道具 举报

15

主题

786

帖子

5

精华

论坛大神

Rank: 7Rank: 7Rank: 7

积分
3223
金钱
3223
注册时间
2015-7-26
在线时间
811 小时
发表于 2015-11-16 16:08:39 | 显示全部楼层
回复【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吧
我的博客:http://blog.csdn.net/itdo_just
回复

使用道具 举报

38

主题

196

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1174
金钱
1174
注册时间
2015-11-9
在线时间
248 小时
 楼主| 发表于 2015-11-17 08:42:53 | 显示全部楼层
回复【7楼】229382777@qq.com:
--------------------------------
你说的这些我都理解,只是有一点不理解#define     GPIOA         ((GPIO_TypeDef *) GPIOA_BASE)这句定义之后为什么结构体中的第一个元素就是0x40010800了(GPIOA_BASE = (APB2PERIPH_BASE + 0x0800) = ERIPH_BASE + 0x10000 + 0x0800 = 0x40000000 + 0x10000 + 0x0800 
= 0x40010800这个基地址我也理解)
回复

使用道具 举报

15

主题

786

帖子

5

精华

论坛大神

Rank: 7Rank: 7Rank: 7

积分
3223
金钱
3223
注册时间
2015-7-26
在线时间
811 小时
发表于 2015-11-17 10:02:30 | 显示全部楼层
首先GPIOA_BASE是常量,我们不能对常量进行操作,所以把它转为指针变量,前面你说根据基地址连续操作寄存器你能懂,那请问为什么能根据基地址进行操作?如果没有结构体,你只能把每个寄存器的地址常量都转为指针变量后再进行操作,但有了结构体后就不需要了,结构体有个特性,结构体的内存地址如果是对齐的话地址是连续的,刚好符合我们的操作需求,而且使这个地址有了范围,就像你把char类型强制转为了int类型,为什么就从1字节变成了4字节是一个道理,结构体代表的是一段内存空间,而你强制转换就是为这段空间取了个名字叫0x40010800,那么自然他的第一个元素就是0x40010800,就像数组名代表他的首地址一样,强烈建议学好C呀,C博大精深,自己也只会皮毛
我的博客:http://blog.csdn.net/itdo_just
回复

使用道具 举报

4

主题

19

帖子

0

精华

初级会员

Rank: 2

积分
68
金钱
68
注册时间
2015-10-12
在线时间
3 小时
发表于 2015-11-17 19:27:52 | 显示全部楼层
回复【6楼】shibinjie:
---------------------------------
是的,是变为了指针,我说的是BASE变为了地址  int *P    *p是该地址内的内容,p就是地址
回复

使用道具 举报

38

主题

196

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1174
金钱
1174
注册时间
2015-11-9
在线时间
248 小时
 楼主| 发表于 2015-11-18 10:03:29 | 显示全部楼层
回复【9楼】229382777@qq.com:
---------------------------------
谢谢解释的很清楚了,越来越觉得自己C没学好了
回复

使用道具 举报

32

主题

230

帖子

0

精华

高级会员

Rank: 4

积分
508
金钱
508
注册时间
2016-4-14
在线时间
113 小时
发表于 2016-6-21 18:07:39 | 显示全部楼层
229382777@qq.co 发表于 2015-11-17 10:02
首先GPIOA_BASE是常量,我们不能对常量进行操作,所以把它转为指针变量,前面你说根据基地址连续操作寄存器 ...

解释的很到位,谢谢
回复

使用道具 举报

32

主题

230

帖子

0

精华

高级会员

Rank: 4

积分
508
金钱
508
注册时间
2016-4-14
在线时间
113 小时
发表于 2016-6-21 18:14:00 | 显示全部楼层
回复

使用道具 举报

2

主题

40

帖子

0

精华

初级会员

Rank: 2

积分
99
金钱
99
注册时间
2021-5-15
在线时间
14 小时
发表于 2021-5-16 21:32:41 | 显示全部楼层
it_do_just 发表于 2015-11-16 12:34
((GPIO_TypeDef *) GPIOA_BASE)   //这里将GPIOA基地址强制转为了GPIO_TypeDef这 ...

GPIO_TypeDef这个名字有好几个结构体全叫这个名字,咋回事啊,蒙
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-5-24 01:51

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表