首页 > 解决方案 > 对 C 宏的未定义引用,但在我定义它时出现重新定义错误

问题描述

我在 kthread.h 中定义了一个宏,如下所示。

#define KERNEL_THREAD_SAVED_KERENL_TOP_OFFSET 208

我正在尝试在 syscall.cc 中使用该宏

#include "syscall.h"
#include "kthread.h"

#define SET_KERNEL_THREAD_TOP_OFFSET(offset, reg) \
  "movq " #offset "(%%" #reg "), %%rsp \n"

// .. and in some function
void func() {
  asm volatile(
    SET_KERNEL_THREAD_TOP_OFFSET(KERNEL_THREAD_SAVED_KERNEL_TOP_OFFSET, rbx)
:::);
}

如果我编译它,它会给出以下链接器错误。

syscall.cc:41: undefined reference to `KERNEL_THREAD_SAVED_KERNEL_TOP_OFFSET'

好的,我明白了。那么可能链接器没有找到宏?所以我也尝试在 .cc 中用一些不同的值来定义它。

#include "syscall.h"
#include "kthread.h"

#define KERNEL_THREAD_SAVED_KERNEL_TOP_OFFSET 100
#define SET_KERNEL_THREAD_TOP_OFFSET(offset, reg) \
  "movq " #offset "(%%" #reg "), %%rsp \n"

然后,我得到一个编译器错误

./kernel/syscall.cc:5: error: "KERNEL_THREAD_SAVED_KERNEL_TOP_OFFSET" redefined [-Werror]
    5 | #define KERNEL_THREAD_SAVED_KERNEL_TOP_OFFSET 100
      | 
In file included from ./kernel/syscall.cc:3:
./kernel/kthread.h:7: note: this is the location of the previous definition
    7 | #define KERNEL_THREAD_SAVED_KERNEL_TOP_OFFSET 208
      | 

很明显,编译器知道 syscall.cc 中的 KERNEL_THREAD_SAVED_KERNEL_TOP_OFFSET。那为什么链接器看不到定义呢??

标签: c++cmacroslinkerc-preprocessor

解决方案


链接器没有业务查找宏。宏由预处理器处理,它在链接器之前运行很长时间。如果链接器抱怨未定义的引用,则意味着您的宏扩展出错了。KERNEL_THREAD_SAVED_KERNEL_TOP_OFFSET它会在不应该出现的地方发射。

问题在于

SET_KERNEL_THREAD_TOP_OFFSET(KERNEL_THREAD_SAVED_KERNEL_TOP_OFFSET, rbx)

这将扩展到

"movq " "KERNEL_THREAD_SAVED_KERNEL_TOP_OFFSET" "(%%" "rbx" "), %%rsp \n"

而您希望它扩展到(使用中的值kthread.h

"movq " "208" "(%%" "rbx" "), %%rsp \n"

它发生的原因是如何处理像宏一样的函数的参数。在 stringification operator 中使用它们时,它们不会进行中间扩展#

您可以通过引入一个间接层来解决它。

#define SET_KERNEL_THREAD_TOP_OFFSET(offset, reg) \
  SET_KERNEL_THREAD_TOP_OFFSET_(offset, reg)
#define SET_KERNEL_THREAD_TOP_OFFSET_(offset, reg) \
  "movq " #offset "(%%" #reg "), %%rsp \n"

现在,当您尝试进行扩展时,它会分几次扩展。首先

SET_KERNEL_THREAD_TOP_OFFSET_(208, rbx)

(rbx不是宏的名称,因此它保持不变)。这然后变成

"movq " "208" "(%%" "rbx" "), %%rsp \n"

推荐阅读