我在 windows10 的 powershell 中使用 gcc 编译器。 gcc 带有 Atollic TrueSTUDIO ide。我这样做的原因是能够从 C 代码创建一个 .exe 文件,以便单元测试变得更容易。
当存在一个定义为弱的函数并且该函数在另一个 .c 文件中使用时,我遇到链接器错误(对“函数名称”的 undefined reference )。
同时,如果我使用 arm-atollic-eabi-gcc 或在 ubuntu 上运行的 gcc,我不会收到此链接器错误。
下面是一个简单的代码来演示这一点:
你好.c:
#include "weak.h"
void whatever(void)
{
iamweak();
}
弱.c:
#include <stdio.h>
#include "weak.h"
void __attribute__((weak)) iamweak(void)
{
printf("i am weak...\n");
}
弱.h
void iamweak(void);
主.c
int main(void)
{
return 0;
}
创建目标文件并链接:
> gcc -c main.c weak.c hello.c
> gcc -o main.exe main.o weak.o hello.o
> hello.o:hello.c:(.text+0x7): undefined reference to `iamweak' collect2.exe: error: ld returned 1 exit status
现在我用gcc-nm检查了hello.o的符号表:
> gcc-nm hello.o
00000000 b .bss
00000000 d .data
00000000 r .eh_frame
00000000 r .rdata$zzz
00000000 t .text
U _iamweak
00000000 T _whatever
weak.o 的符号表:
>gcc-nm weak.o
00000000 b .bss
00000000 d .data
00000000 r .eh_frame
00000000 r .rdata
00000000 r .rdata$zzz
00000000 t .text
00000000 T .weak._iamweak.
w _iamweak
U _puts
现在,当我在 Ubuntu 上使用 gcc 时,一切正常。符号表也略有不同。
hello.o 的符号表:
nm hello.o
U _GLOBAL_OFFSET_TABLE_
U iamweak
0000000000000000 T whatever
weak.o 的符号表:
nm weak.o
U _GLOBAL_OFFSET_TABLE_
0000000000000000 W iamweak
U puts
来自 https://linux.die.net/man/1/nm它说“如果小写,符号是局部的;如果大写,符号是全局的(外部的)。”
所以iamweak在windows10是本地的,在Ubuntu是全局的。这就是链接器看不到它的原因吗?我该怎么办?弱函数定义也在一些 HAL 库中,我不想修改它们。有解决方法吗?
最佳答案
这是 atollic gcc fork 错误。情况更糟:
main:
00401440: push %ebp
00401441: mov %esp,%ebp
00401443: and $0xfffffff0,%esp
00401446: call 0x401970 <__main>
36 iamweak();
0040144b: call 0x0
37 return 0;
00401450: mov $0x0,%eax
38 }
完整的环礁工作室项目 here
关于c - gcc 弱属性不一致行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59963701/