首页 > 解决方案 > gcc 弱属性不一致的行为

问题描述

我在 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 库中,我不想修改它们。有解决方法吗?

标签: cgcclinkertruestudio

解决方案


这是 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        }

完整的环礁工作室项目在这里


推荐阅读