链接描述文件:将函数的绝对地址插入到生成的代码中

问题描述 投票:0回答:2

我有一个与 gcc 链接器相关的问题。

我正在使用嵌入式东西(PIC32),但PIC32的编译器和链接器基于gcc,因此,“常规”gcc链接器和PIC32链接器的主要内容应该是通用的。

为了节省闪存空间(这在微控制器上通常是不够的),我需要在引导加载程序中放置几个大函数,应用程序应该仅通过指针调用这些函数。

所以,我需要在生成的代码中创建向量表。

我试图获取函数的绝对地址并将其写入生成的代码,但仍然出现错误。

在下面的示例中,我试图获取函数的地址

_formatted_write

代码:

.user_addr_table _USER_ADDR_TABLE_ADDR :
{
   KEEP(*(.user_addr_table))
   LONG((ABSOLUTE(_formatted_write))); /* _formatted_write is a name of my function */
} > user_addr_table

链接器返回错误:“

unresolvable symbol '_formatted_write' referenced in expression
”。

我注意到,如果我写一些垃圾而不是

_formatted_write
,那么它会返回错误“
undefined symbol .....
”,因此,链接器知道
_formatted_write

这让我觉得我应该执行一些额外的步骤来使其“可解析”。但还是不知道该怎么做。

c gcc ld linker-scripts
2个回答
2
投票

对于有人有同样问题的情况(就像我几个小时前一样),还有一个更好的解决方案:

让“bootloader.out”成为引导加载程序二进制文件。然后我们可以用

生成
nm -g bootloader.out | grep '^[0-9,a-f,A-F]\{8\} T' | awk '{printf "PROVIDE(%s = 0x%s);\n",$3,$1}' > bootloader.inc

我们可以将其包含在应用程序的链接描述文件中的文件

INCLUDE bootloader.inc

链接器现在知道所有引导加载程序函数的地址,我们可以链接它,而无需在应用程序中包含实际的函数代码。我们所需要的只是我们想要执行的每个引导加载程序函数的声明。


1
投票

我发现解决这个问题的方法是使用函数指针,即

void (*formatted_write)( int a, char * b, struct mystruct c );

然后,在启动时的代码中设置该函数的地址,即

formatted_write = 0xa0000;

然后您可以使用正常约定调用您的函数。

此外,您可以在编译时使用 -Wl,--just-symbols=symbols.txt 标志。您可以按如下方式设置符号的地址:

formatted_write = 0xa0000;
© www.soinside.com 2019 - 2024. All rights reserved.