c - gcc 弱属性不一致行为

标签 c gcc linker truestudio

我在 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/

相关文章:

c++ - 如何在 C/C++ 中临时禁用宏扩展?

c++ - 整个项目的 GCC 选项 -ftime-report 摘要

C++ 链接器问题,是否有解决这些问题的通用方法?

c - for 循环比较中什么更快?

C <pthread.h> 将局部变量传递给线程已损坏

c - 估计代码在本地linux系统上的执行时间

c++ - emscripten ASSERTIONS=1 生成链接器错误 "Referencing global in another module!"

c - 使用 memset 和 int 值初始化整数数组 - 失败

c - KPIT GCC 将文本字符串分配给特定的链接器部分

ios - 在iOS上链接libiconv