繁体   English   中英

如何在 C++ 宏中包含一个换行符或如何使用 C++ 模板来做同样的事情?

[英]How to include a newline in a C++ macro or how to use C++ templates to do the same?

我看到以下问题: How to generate a newline in a cpp macro?

让我简要介绍一下 C++ 预处理器中换行的需求。 我正在研究 ARM Realview 编译器 3.1,该代码使用带有 C++ 代码的嵌入式汇编代码。

#define DEFINE_FUNCTION(rtype, op, val) \
    __asm rtype nt_##op(void*) { \   
      str lr, [sp, ##val];bl vThunk_##op;ldr lr, [sp,##val];bx lr; \
    } \
   void vThunk_##op(void*)

DEFINE_FUNCTION(void*, mov_lt, #0x04)
{
     // do some operation
}

上面的宏声明了一个嵌入式程序集 function,它强制要求 function 主体中的每个语句之间换行。

我认为这是因为 function 主体中的文本被 ARM 编译器盲目发送给 ARM 汇编器。

为什么 C++ 预处理器现在仍然不支持多行替换? 而且我不能在替换字符串中使用#。 例如,对于这种装配,

str lr, [sp, #0x04]

我尝试了很多方法和方法,但没有真正奏效。 ARM汇编器/编译器太基础了,GCC中没有像asm volatile这样的API。

DEFINE_FUNCTION 宏用在很多地方,所以也不能忽略它。

因此,最终考虑以下解决方案:

  • 使用 m4 预处理器代替 C++ 预处理器
  • 使用 C++ 模板以某种方式实现此目的并使用 grep/sed 替换 DEFINE_FUNCTION

任何人都可以给我指示或方法来做上述事情吗? 除了 ARM Realview 编译器 3.1,我不能使用任何编译器。

我需要像下面这样的新行进行一些扩展, DEFINE_FUNCTION(void*, mov_lt, #0x04) {}

__asm void* nt_mov_lt(void*) {   
      str lr, [sp, 0x04];
      bl vThunk_mov_lt;
      ldr lr, [sp,0x04];
      bx lr;
    }
 void vThunk_mov_lt(void*)
 {
     // do something
  }

我成功地使用GNU m4预处理器解决了上述问题。

m4_define('DEFINE_FUNCTION','
     __asm rtype nt_$2(void*) {  
      str lr, [sp, $3];
      bl vThunk_$1;
      ldr lr, [sp,$3];
      bx lr;
    } 
   void vThunk_$2(void*)')

DEFINE_FUNCTION(void*, mov_lt, 0x04)
{
}

使用m4预处理上面的代码解决了代码中的换行符问题。 将m4 -P作为预建事件,以便在C预处理器和编译阶段进入画面之前处理源文件。

感谢您的帮助,抱歉让您感到困惑。 但是在最新的C ++编译器中确实存在良好的宏预处理器的范围。

每个气体版本都有不同的换行替换字符,例如AVR有$寻找ARM气体版本的字符

你的例子非常令人困惑,但这不会起作用吗?

#define DEFINE_FUNCTION(rtype, op, val) \
    __asm rtype nt_##op(void*) { \   
      str lr, [sp, val];\
      bl vThunk_##op;\
      ldr lr, [sp,val];\
      bx lr;\
    }\
   void vThunk_##op(void*)

当被称为时

DEFINE_FUNCTION(void*, mov_lt, 0x04)  // notice no # here
{
     // do some operation
}

你会得到的

__asm void* nt_mov_lt(void*) {   
      str lr, [sp, 0x04];
      bl vThunk_mov_lt;
      ldr lr, [sp,0x04];
      bx lr;
    }
 void vThunk_mov_lt(void*)
 {
     // do something
 }

这正是你所要求的。

我不知道 ARM 的各种 C 和 C++ 编译器,但以下内容至少在 2022 年适用于 Windows x86 / x64 的 Microsoft 编译器,以及 Open Watcom 编译器(甚至适用于 DOS):

#define MY_MACRO { \
  _asm { \
    _asm MOV AX, 0 \
    _asm MOV BX, 1 \
  } \
}

预处理器将其变成一行,删除连续字符和换行符,但每个_asm标记将指令分隔开,就好像有一个换行符一样。

再次强调一下,我对 ARM 毫无头绪,而且 GCC x86 / x64 的语法无论如何都不一样,但也许你可以将这个想法应用到你的问题中。

PS 我知道最外面的花括号是多余的,但这是我在编写这样的宏时更喜欢的方式。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM