美文网首页C语言C语言笔记
【C语言学习】带参宏定义(一)

【C语言学习】带参宏定义(一)

作者: 学以解忧 | 来源:发表于2019-08-29 16:58 被阅读0次

先看一个带参宏的例子:

image

也即:

#define FPGA_WRITE(data_out, base_addr, offset)  \
((((volatile uint32*)base_addr)[(offset)]) = (data_out))
#define FPGA_READ(data_in, base_addr, offset)    \
((data_in) = (((volatile uint32*)base_addr)[(offset)]))

带参宏也称做宏函数,宏函数FPGA_WRITE(data_out, base_addr, offset)用于往FPGA发送数据。其中

  • 参数data_out:要写入的数据。

  • 参数base_addr:基地址。

  • 参数offset:相对于基地址的偏移量。

该宏函数实体为:

((((volatile uint32*)base_addr)[(offset)]) = (data_out))

我们先看等号左边,从最里层的括号开始看,(volatile uint32*)base_addr的意思是把base_addr强制转换为(volatile uint32*)类型的数据,其中加volatile关键字的作用是确保本条指令不会因编译器的优化而省略。volatile在嵌入式编程中用得很多,如在Cortex-M3内核MCU的内核文件的C函数内嵌汇编中使用了大量的volatile关键字:

image

关于volatile关键字更多的介绍可查看往期分享:
【C语言笔记】volatile关键字

((volatile uint32*)base_addr)[(offset)]的意思是相对于base_addr偏移offset个内存单元后所占的空间,此时已经转变为了base_addr[offset],这就可以看做数组base_addr的第offset个元素,所以可以给它赋值。等号右边的数据data_out就是给数组元素base_addr[offset]进行赋值的。数组和指针在此处其实是可以等价的,所以

((volatile uint32*)base_addr)[(offset)]

其实可以等价为:

*(((volatile uint32*)base_addr) + offset)

宏函数FPGA_READ(data_in, base_addr, offset)用于读取FPGA发送过来的数据,其实体为:

((data_in) = (((volatile uint32*)base_addr)[(offset)]))

同写函数的分析方法类似,此处从等号右边的最里层括号开始看,分析过程省略,具体的可查看写数据函数FPGA_WRITE的分析过程。此处等号右边的

(((volatile uint32*)base_addr)[(offset)]))

可以等价于:

*(((volatile uint32*)base_addr) + offset))

为什么要在宏函数实体的参数的两边加上括号呢?为什么要在宏函数实体的两边加上括号呢?

答:虽然有时候不加括号也没什么问题,但是,更严格的做法是给参数加括号、给宏函数实体加括号,这样可以避免二义性。关于宏函数的二义性将在下一篇笔记中分享,欢迎阅读!

以上两个带参宏的测试用例

/********************************************************************************
* 宏函数FPGA_WRITE、FPGA_READ测试用例
******************************************************************************/
#include <stdio.h>

#define uint32 unsigned int

#if 1
// 调用这两个宏可往共享内存中读写数据
#define FPGA_WRITE(data_out, base_addr, offset) 
\((((volatile uint32*)base_addr)[(offset)]) = (data_out))
#define FPGA_READ(data_in, base_addr, offset) 
\((data_in) = (((volatile uint32*)base_addr)[(offset)]))

#else
// 以上宏等效的写法
#define FPGA_WRITE(data_out, base_addr, offset) 
\(*(((volatile uint32*)base_addr) + offset) = (data_out))
#define FPGA_READ(data_in, base_addr, offset) 
\((data_in) = *(((volatile uint32*)base_addr) + offset))
#endif

int main(void)
{
  // 变量定义
  uint32 arr[6] = {0, 1, 2, 3, 4, 5};
  uint32 *ptr = arr;
  uint32 data_write = 520;
  uint32 data_read = 0;

  // 验证宏函数FPGA_WRITE
  FPGA_WRITE(data_write, ptr, 3); // 此时arr[3]的值会被data_write的值覆盖
  printf("arr[3] = %d\n", arr[3]);
  if (data_write == arr[3])
  {
    printf("宏函数FPGA_WRITE验证成功!\n\n");
  }

  // 验证宏函数FPGA_READ
  FPGA_READ(data_read, ptr, 5); // 此时data_read的值会被arr[5]的值覆盖
  printf("data_read = %d\n", data_read);
  if (data_read == arr[5])
  {
    printf("宏函数FPGA_READ验证成功!\n");
  }
  return 0;
}

以上的测试方法是:定义一个数组arr,定义一个基地址ptr(指针变量),基地址ptr指向arr,此时ptr就可以与数组arr相关联起来了,即相对于ptr偏移offset个内存单元其实就是等价于arr[offset]。

FPGA_WRITE(data_write, ptr, 3);

这条语句的意思就是往ptr往后第3个内存单元写入数据data_write,即arr[3] = data_write;,arr[3]由原来的3变成了520。

FPGA_READ(data_read, ptr, 5);

这条语句的意思就是把ptr往后第5个内存单元中的数据赋给data_read变量,即data_read = arr[5];,data_read由原来的0变成了5。程序运行结果如下:


image

可见,程序输出结果与我们分析的一致!带参宏很重要,在一定程度上可以帮助我们防止出错,提高代码的可移植性和可读性等,应重点掌握。下一篇笔记我们将分享更多的带参宏的笔记,欢迎阅读。


关注我的微信公众号【嵌入式大杂烩】,回复:C语言,可获取C语言资料。
关注我的个人博客【我的博客】查看更多笔记

相关文章

  • C语言-宏定义-带参宏

    宏定义指令 (# define)用来定义一个标识符和一个字符串,以这个标识符来代表这个字符串。 不带参数的宏定义:...

  • 【C语言学习】带参宏定义(一)

    先看一个带参宏的例子: 也即: 带参宏也称做宏函数,宏函数FPGA_WRITE(data_out, base_ad...

  • 【C语言学习】带参宏定义(二)

    带参宏在我们的嵌入式编程中使用得非常多,其定义如下: 其中参数列表中的参数之间用逗号分隔,字符序列中应包含参数表中...

  • 精解C语言预处理命令(三)之“宏”的用法二

    关于C语言带参数的宏定义中的参数 C语言允许宏带有参数。在宏定义中的参数称为“形式参数”,在宏调用中的参数称为“实...

  • 宏定义与常量的基本用法

    宏定义 宏定义分为两种:带参定义与无参定义 带参定义 无参数定义 常量定义 常量定义也分为两种:全局与局部(相对整...

  • 10/19

    今天老师讲了预处理命令,宏定义分为无参宏定义,带参宏定义和条件编译。宏定义包括宏名和宏展开,和函数相比预处理有很多...

  • iOS - 《系统宏》宏,预处理命令

    在ios中使用预处理命令[C语言的特殊命令] 1.宏定义 #define 与 #undef 2.带参数宏定义 3....

  • C语言学习:C语言宏定义

    学C语言很久了,但还是不敢用宏定义,大神喜欢用宏定义。关于宏定义你又了解多少了,下面我们说一下宏定义。 宏定义的概...

  • C++<第十一篇>:宏定义define用法

    define分为无参宏定义和有参宏定义 一、无参宏定义 无参宏的宏名后不带参数。 其定义的一般格式为: # 表示预...

  • 总结

    宏定义:宏定义的分为无参宏定义与有参宏定义。无参宏定义的一般形式为:#define 标识符 字符串。‘#’表示...

网友评论

    本文标题:【C语言学习】带参宏定义(一)

    本文链接:https://www.haomeiwen.com/subject/qktwectx.html