我在 windows10 的 powershell 中使用 gcc 编译器。gcc 带有 Atollic TrueSTUDIO ide。我这样做的原因是能够从 C 代码创建一个 .exe 文件,以便单元测试变得更容易。
当有一个函数被定义为弱并且该函数在另一个 .c 文件中使用时,我遇到了链接器错误(未定义的对“function_name”的引用)。
同时,如果我使用在 ubuntu 上运行的 arm-atollic-eabi-gcc 或 gcc,我不会收到此链接器错误。
这是一个简单的代码来演示这一点:
你好ç:
#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);
主程序
int main(void)
{
return 0;
}
创建目标文件和链接:
> gcc -c main.c weak.c 你好.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 吨 .text
u_iamweak
00000000 T _whatever
weak.o 的符号表:
>gcc-nm 弱.o
00000000 b .bss
00000000 d .data
00000000 r .eh_frame
00000000 r .rdata
00000000 r .rdata$zzz
00000000 吨 .text
00000000 T .weak._iamweak。
w _iamweak
U_puts
现在,当我在 Ubuntu 上使用 gcc 时,我说一切正常。符号表也有些不同。
hello.o 的符号表:
纳米你好.o
U _GLOBAL_OFFSET_TABLE_
你是弱者
0000000000000000 T 不管怎样
weak.o 的符号表:
纳米弱.o
U _GLOBAL_OFFSET_TABLE_
0000000000000000 威弱
U看跌
从https://linux.die.net/man/1/nm它说“如果是小写,则符号是本地的;如果是大写,则符号是全局的(外部)。”
所以 iamweak 在 windows10 中是本地的,在 Ubuntu 中是全局的。这就是链接器看不到它的原因吗?我能做些什么呢?弱函数定义也在一些 HAL 库中,我不想修改它们。有解决方法吗?