我可以将普通文件链接到我的可执行文件吗?

时间:2012-12-13 09:36:54

标签: c++ c gcc linker embedded-resource

某些框架(Qt,Windows,Gtk ...)提供了向二进制文件添加资源的功能。我想知道是否有可能在没有框架的情况下实现这个,因为所有真正需要的是

  1. 包含二进制(数据段)中资源地址的符号
  2. 表示资源长度的符号
  3. 资源本身
  4. 如何使用gcc工具链实现这一目标?

2 个答案:

答案 0 :(得分:48)

你可以这样做:

objcopy --input binary \
        --output elf32-i386 \
        --binary-architecture i386 my_file.xml myfile.o

这会生成一个可以链接到可执行文件的目标文件。 此文件将包含您必须在C代码中声明的这些符号 能够使用它们

00000550 D _binary_my_file_xml_end
00000550 A _binary_my_file_xml_size 
00000000 D _binary_my_file_xml_start

答案 1 :(得分:31)

最基本的,等价物是一个充满字节的char数组。

在Linux上,您可以使用xxd -i <file>将文件“编译”到char数组中,然后将数组链接到二进制文件中,然后使用组成字节。

以下是我自己的代码makefile的示例,它创建了一个名为templates.h的“资源文件”,其中包含一堆表示HTML模板的char数组:

templates.h:
    @echo "#ifndef REDACTED_TEMPLATES_H" > templates.h
    @echo "#define REDACTED_TEMPLATES_H" >> templates.h
    @echo "// Auto-generated file! Do not modify!" >> templates.h
    @echo "// NB: arrays are not null-terminated" >> templates.h
    @echo "// (anonymous namespace used to force internal linkage)" >> templates.h
    @echo "namespace {" >> templates.h
    @echo "namespace templates {" >> templates.h
    @cd templates;\
    for i in * ;\
    do \
        echo "Compiling $$i...";\
        xxd -i $$i | sed -e 's/ =/ __attribute__((unused)) =/' >> ../templates.h;\
    done;\
    cd ..
    @echo "}" >> templates.h
    @echo "}" >> templates.h
    @echo "#endif" >> templates.h

另见: How best can I programmatically apply `__attribute__ ((unused))` to these auto-generated objects?

结果看起来有点像:

#ifndef REDACTED_TEMPLATES_H
#define REDACTED_TEMPLATES_H
// Auto-generated file! Do not modify!
// NB: arrays are not null-terminated
// (anonymous namespace used to force internal linkage)
namespace {
namespace templates {
unsigned char alert_email_finished_events_html[] __attribute__((unused)) = {
  0x3c, 0x74, 0x61, 0x62, 0x6c, 0x65, 0x20, 0x63, 0x6c, 0x61, 0x73, 0x73,
  0x3d, 0x22, 0x6e, 0x6f, 0x64, 0x65, 0x2d, 0x69, 0x6e, 0x66, 0x6f, 0x2d,
[..]
  0x7d, 0x7d, 0x0d, 0x0a, 0x3c, 0x2f, 0x74, 0x61, 0x62, 0x6c, 0x65, 0x3e,
  0x0d, 0x0a
};
unsigned int alert_email_finished_events_html_len __attribute__((unused)) = 290;
unsigned char alert_email_finished_events_list_html[] __attribute__((unused)) = {
  0x3c, 0x74, 0x72, 0x20, 0x63, 0x6c, 0x61, 0x73, 0x73, 0x3d, 0x22, 0x73,
  0x65, 0x70, 0x61, 0x72, 0x61, 0x74, 0x65, 0x2d, 0x70, 0x72, 0x65, 0x76,
[..]
  0x73, 0x74, 0x7d, 0x7d, 0x0d, 0x0a
};
unsigned int alert_email_finished_events_list_html_len __attribute__((unused)) = 42;
}
}
#endif

请注意,在仅使用一个翻译单元中的资源时,此特定示例是最佳的,但可以调整一般方法以满足您的需求。