从二进制文件中提取数组数据的正确方法?

时间:2015-10-28 22:53:43

标签: c++ c file

有一种经典的方法可以将资源文件作为C语言数组嵌入到二进制文件中,这样我们就可以将一些外部资源文件(如.jpeg.txt文件)存储到二进制文件中。 / p>

例如,在头文件中我们可以定义一个数组:

const unsigned char xd_data[] = {
    77,90,144,0,3,0,0,0,4,0,0,0,255,255,0,0,184,0,0,0,0,0,0,0,64,0,0,0,0,
    0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,240,0,0,
    0,14,31,186,14,0,180,9,205,33,184,1,76,205,33,84,104,105,115,32,112,114,
    111,103,114,97,109,32,99,97,110,110,111,116,32,98,101,32,114,117,110,
    32,105,110,32,68,79,83,32,109,111,100,101,46,13,13,10,36,0,0,0,0,0,0,
    0,66,163,223,218,6,194,177,137,6,194,177,137,6,194,177,137,105,221,187,
    137,13,194,177,137,133,222,191,137,3,194,177,137,105,221,181,137,4,194,
    177,137,136,202,238,137,4,194,177,137,6,194,176,137,73,194,177,137,133,
    202,236,137,13,194,177,137,48,228,187,137,11,194,177,137,193,196,183,
    137,7,194,177,137,82,105,99,104,6,194,177,137,0,0,0,0,0,0,0,0,0,0,0,0,
    0,0,0,0,0,0,0,0,0,0,0,0,80,69,0,0,76,1,4,0,65,162,32,86,0,0,0,0,0,0,0,
    0,224,0,47,1,11,1,6,0,0,100,0,0,0,74,0,0,0,0,0,0,228,113,0,0,0,16,0,0,
    0,128,0,0,0,0,64,0,0,16,0,0,0,2,0,0,4,0,0,0,0,0,0,0,4,0,0,0,0,0,0,0,0,
    224,0,0,0,4,0,0,0,0,0,0,2,0,0,0,0,0,16,0,0,16,0,0,0,0,16,0,0,16,0,0,0,
    0,0,0,16,0,0,0,0,0,0,0,0,0,0,0,124,140,0,0,140,0,0,0,0,208,0,0,0,16,0
};

包含资源文件的内容,它将编译成最终的二进制文件。

网上有很多关于这个老技巧的工具和教程,例如:http://www.rowleydownload.co.uk/arm/documentation/index.htm?http://www.rowleydownload.co.uk/arm/documentation/embed.htmhttps://www.fourmilab.ch/xd/http://gareus.org/wiki/embedding_resources_in_executables#c_include_method

然而,看起来大多数这些页面都在谈论如何使用C样式数组将数据嵌入到二进制文件中。

我的问题是,在编译的二进制文件中找到资源文件的起始地址以便提取它们的正确方法是什么?即,如何在已编译的二进制文件中找到xd_data的起始地址?

2 个答案:

答案 0 :(得分:1)

如果您的意思是在文件中找到数据块的字节地址,就像objdump那样,但是以编程方式进行,那么您可以使用二进制文件描述符库(BFD),请参阅here和{ {3}}

答案 1 :(得分:0)

如果您存储了数据,例如图像,并且您想要加载它(用于打印或任何您想要的),那么如果您有一个从内存加载它的函数(库),例如void loadResImage(void * mem);就行了loadResImage(xd_data),如果没有,但你有一个从文件加载它的函数,在这种情况下将它保存到临时文件,例如:

int fd=open("tmpfile");
int ret=write(fd,xd_data, sizeof(xd_data));
close(fd);
loadImageFile("tmpfile");

但是如果你想访问程序本身之外的数据(例如十六进制编辑器或其他程序),那么你必须添加一个起始标记和一个结束标记或sizeof数据。例如:

const unsigned char xd_data[]={
  ...
'M','A','G','I','C'};

在上面的示例中,数据结尾已知,您只需进行搜索即可找到它。以同样的方式,玩并找到一种合适的方式来存储数据的大小。但要注意编译器的优化。