OpenEdv-开源电子网

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

宏定义的if问题

[复制链接]

9

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
140
金钱
140
注册时间
2018-2-3
在线时间
17 小时
发表于 2018-2-11 16:12:35 | 显示全部楼层 |阅读模式
1金钱
起因我是想通过一个参数GPIOA 类型(GPIO_TypeDef *)  来确定RCC_APB2Periph_GPIOA
也就是说,通过一个GPIO_TypeDef* GPIOx参数,就可以确定时钟的RCC_APB2Periph,
所以就定义了一下宏定义
#define RCC_APB2Periph_GPIO(GPIOx)                              if(GPIOA == GPIOx)                RCC_APB2Periph_GPIOA

执行函数是这样子的
void GPIO_Config(GPIO_TypeDef* GPIOx,uint8_t GPIO_Pin,GPIOSpeed_TypeDef GPIO_Speed,GPIOMode_TypeDef GPIO_Mode)
{
        GPIO_InitTypeDef GPIO_InitStruct;

        RCC_APB2PeriphClockCmd(         RCC_APB2Periph_GPIO(GPIOx)         ,ENABLE);  //宏定义在这里使用

        GPIO_InitStruct.GPIO_Mode  = GPIO_Mode;
        GPIO_InitStruct.GPIO_Pin   = GPIO_Pin;
        GPIO_InitStruct.GPIO_Speed = GPIO_Speed;       
        GPIO_Init(GPIOx,&GPIO_InitStruct);
}


这样子会报错..\Drivers\GPIO\GPIO.c(21): error:  #29: expected an expression

问题1:我的规格有错误吗?
问题2:可以用宏定义达到我的目的吗?怎么做?

正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

6

主题

462

帖子

0

精华

高级会员

Rank: 4

积分
906
金钱
906
注册时间
2017-12-15
在线时间
111 小时
发表于 2018-2-11 16:18:24 | 显示全部楼层
你的格式是不正确的,并且,它无法满足你的要求。因为宏定义是程序运行前对代码进行的一些初始设定(常见于进行文本替换)。但是,你的代码中,GPIOx是一个变化的值,是由GPIO_configd的参数传入的。如果需要应当这样的变化,必须通过程序,而不是程序编译前的宏定义。
回复

使用道具 举报

1

主题

4

帖子

0

精华

新手上路

积分
24
金钱
24
注册时间
2018-2-11
在线时间
1 小时
发表于 2018-2-11 20:51:55 | 显示全部楼层
这个带参宏需要这样定义#define RCC_APB2Periph_GPIO(GPIOx)       if(GPIOA == GPIOx)RCC_APB2Periph_GPIOA   ; 后面不能有空格
回复

使用道具 举报

1

主题

4

帖子

0

精华

新手上路

积分
24
金钱
24
注册时间
2018-2-11
在线时间
1 小时
发表于 2018-2-11 20:52:27 | 显示全部楼层
TanSuoShiJian 发表于 2018-2-11 20:51
这个带参宏需要这样定义#define RCC_APB2Periph_GPIO(GPIOx)       if(GPIOA == GPIOx)RCC_APB2Periph_GPIO ...

没有分号
回复

使用道具 举报

9

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
140
金钱
140
注册时间
2018-2-3
在线时间
17 小时
 楼主| 发表于 2018-2-11 22:16:08 | 显示全部楼层

加了分号也是一样的RCC_APB2Periph_GPIO 这句也只是个常量
回复

使用道具 举报

9

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
140
金钱
140
注册时间
2018-2-3
在线时间
17 小时
 楼主| 发表于 2018-2-11 22:19:01 | 显示全部楼层
a496298685 发表于 2018-2-11 16:18
你的格式是不正确的,并且,它无法满足你的要求。因为宏定义是程序运行前对代码进行的一些初始设定(常见于 ...

但是函数一调用,参数不就是确定下来了吗?不就可以用宏替换?
回复

使用道具 举报

6

主题

462

帖子

0

精华

高级会员

Rank: 4

积分
906
金钱
906
注册时间
2017-12-15
在线时间
111 小时
发表于 2018-2-11 22:54:50 | 显示全部楼层
赤诚 发表于 2018-2-11 22:19
但是函数一调用,参数不就是确定下来了吗?不就可以用宏替换?

函数是在运行时被调用的,宏只在编译时确定一些内容。
回复

使用道具 举报

9

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
140
金钱
140
注册时间
2018-2-3
在线时间
17 小时
 楼主| 发表于 2018-2-12 00:53:30 | 显示全部楼层
a496298685 发表于 2018-2-11 22:54
函数是在运行时被调用的,宏只在编译时确定一些内容。

也就是说,宏定义可以传给函数做确定参数,确定参数传不回宏定义了?
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165536
金钱
165536
注册时间
2010-12-1
在线时间
2117 小时
发表于 2018-2-19 23:23:17 | 显示全部楼层
RCC_APB2Periph_GPIOA 就是个未定义的宏,你这个写法不对啊
回复

使用道具 举报

9

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
140
金钱
140
注册时间
2018-2-3
在线时间
17 小时
 楼主| 发表于 2018-3-19 00:23:48 | 显示全部楼层
正点原子 发表于 2018-2-19 23:23
RCC_APB2Periph_GPIOA 就是个未定义的宏,你这个写法不对啊

原子哥,#define RCC_APB2Periph_GPIOA             ((uint32_t)0x00000004)这个是官方库文件的定义。不知道在这里能不能直接拿出来用
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165536
金钱
165536
注册时间
2010-12-1
在线时间
2117 小时
发表于 2018-3-20 01:58:57 | 显示全部楼层
赤诚 发表于 2018-3-19 00:23
原子哥,#define RCC_APB2Periph_GPIOA             ((uint32_t)0x00000004)这个是官方库文件的定义。不 ...

如果有定义,就可以。没有就自己定义一下。
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

10

主题

196

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
390
金钱
390
注册时间
2018-3-20
在线时间
80 小时
发表于 2018-3-20 14:05:44 | 显示全部楼层
本帖最后由 孟亮 于 2018-3-20 14:13 编辑

#define _SetEPAddress(bEpNum,bAddr) _SetENDPOINT(bEpNum,\
    EP_CTR_RX|EP_CTR_TX|(_GetENDPOINT(bEpNum) & EPREG_MASK) | bAddr)

上面是定义,下面是引用,感觉会对你有帮助

for (i = 0; i < nEP; i++)
  {
    _SetEPAddress((uint8_t)i, (uint8_t)i);
  } /* for */


可以宏定义个函数,甚至传参数,但是C语言里不支持变量类型的判断,只能C++以上才有。。。所以你这个按类型判断不行,但是可以按名称匹配,把所有可能的参数都写出来,人工判断类型。那样更费事。。。
回复

使用道具 举报

10

主题

196

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
390
金钱
390
注册时间
2018-3-20
在线时间
80 小时
发表于 2018-3-20 14:20:21 | 显示全部楼层
uint32_t AutoRCC(GPIO_TypeDef* GPIOx)
{
        if(GPIOx==GPIOA) return RCC_APB2Periph_GPIOA;
        if(GPIOx==GPIOB) return RCC_APB2Periph_GPIOB;
        //………………
}
非要实现只能这么干了。。。
回复

使用道具 举报

9

主题

28

帖子

0

精华

初级会员

Rank: 2

积分
140
金钱
140
注册时间
2018-2-3
在线时间
17 小时
 楼主| 发表于 2018-3-20 23:02:41 | 显示全部楼层
孟亮 发表于 2018-3-20 14:05
#define _SetEPAddress(bEpNum,bAddr) _SetENDPOINT(bEpNum,\
    EP_CTR_RX|EP_CTR_TX|(_GetENDPOINT(bEp ...

大哥大哥,我不是很懂这个,能稍微点一下吗,你下一条贴子我是看懂了,上面的看不懂
回复

使用道具 举报

10

主题

196

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
390
金钱
390
注册时间
2018-3-20
在线时间
80 小时
发表于 2018-3-21 10:33:52 | 显示全部楼层
赤诚 发表于 2018-3-20 23:02
大哥大哥,我不是很懂这个,能稍微点一下吗,你下一条贴子我是看懂了,上面的看不懂

上面这条的意思是 你用define的话,可以带参数,不过前提是你先声明一个函数被define用,实际的结果是编译器帮你 ctrl C, ctrl V到代码里,所以把你要定义的直接贴过去,看看报错和实际代码,就能明白了。你的例子是,直接把(&nbsp;if(GPIOA == GPIOx) &nbsp; &nbsp; &nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp;&nbsp;RCC_APB2Periph_GPIOA)放进程序代码里,变成RCC_APB~(if(GPIO==~)RCC_APB2Per~)自然是会被报错的,rcc函数需要个参数,你把一个判断语句放进去怎么对。
非要用#的话,使用#if 让编译器帮你替换 倒是可以。
回复

使用道具 举报

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

本版积分规则



关闭

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

正点原子公众号

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

GMT+8, 2025-6-10 05:14

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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