第一章.什么是map文件
map文件是编译器编译之后生成的,集函数、数据及IO空间的一种映射文件。
在遇到内存越界或溢出的情况,首先想到的就是分析map文件。通过map文件可以知道变量大小、变量地址、函数入口地址等一些重要信息。
第二章.利用map查找问题
1. 用文本编辑器打开map文件
可以看到每个函数,每个变量的地址、大小等等信息
2. 利用map查找变量值异常问题
2.1 构造一个简单的例子
两个变量和一个数组,在memset故意对array1越界清零,执行结果如下
可以看到结果,var1没有被清零,反而是排在array1前面的var2被清零了
2.2 根据map文件查找问题
打开map文件,搜索var2
var2是int类型,起始地址是0x000000000011e794,大小是4个字节
因为是低位在前,所以内存中0x000000000011e794- 0x000000000011e797实际保存的是 0x78 0x56 0x34 0x12
array1的起始地址是0x000000000011e394,只有1024。memset清除了1025,多清零了1个字节,0x78被错误置为0,所以var2变为0x12345600
2.3 结论
在map文件中搜索变量名,找到它的地址
观察在这个地址之前有没有数组之类容易越界操作的变量
在程序中搜索所有与这个数组有关的代码,确认每次写入都对下标进行了判断
第三章.那么如何在Vitis/SDK中生成map文件
1. 创建一个空文件用于保存map
新建一个空的文本文件,我这里完整路径是这样
c:\debug\test.map
2. 在项目名称上右键,选择Properties(注意不要误选上面的system,要选app)
3. 左边选C/C++ Build的Settings,然后选ARM R5 gcc linker的Miscellaneous
关键是gcc linker,不同的app对应的不一定是arm r5,可能是其他值。
map是属于链接阶段生成的文件
在Linker Flags中一般已经有一些参数了,保留这些参数不变,在最后加上
-Wl,-Map c:\debug\test.map
注意W后面是小写的L, 代表linker的意思
Wl后面的东西是作为参数传递给链接器ld,Wl后面必须用逗号分隔
gcc -Wl,-Map c:\debug\test.map
实际的含义是
ld -Map c:\debug\test.map
4. 然后重新build整个工程
就会看到文件大小从0变成了几百k