• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# 内存泄漏检测<a name="ZH-CN_TOPIC_0000001079076672"></a>
2
3-   [基础概念](#section1026719436293)
4-   [功能配置](#section13991354162914)
5-   [开发指导](#section95828159308)
6    -   [开发流程](#section369844416304)
7    -   [编程实例](#section460801313313)
8    -   [示例代码](#section96539275311)
9    -   [结果验证](#section20527343183119)
10
11
12## 基础概念<a name="section1026719436293"></a>
13
14内存泄漏检测机制作为内核的可选功能,用于辅助定位动态内存泄漏问题。开启该功能,动态内存机制会自动记录申请内存时的函数调用关系(下文简称LR)。如果出现泄漏,就可以利用这些记录的信息,找到内存申请的地方,方便进一步确认。
15
16## 功能配置<a name="section13991354162914"></a>
17
181.  LOSCFG\_MEM\_LEAKCHECK:开关宏,默认关闭;若打开这个功能,在target\_config.h中将这个宏定义为1。
192.  LOSCFG\_MEM\_RECORD\_LR\_CNT:记录的LR层数,默认3层;每层LR消耗sizeof\(void \*\)字节数的内存。
203.  LOSCFG\_MEM\_OMIT\_LR\_CNT:忽略的LR层数,默认4层,即从调用LOS\_MemAlloc的函数开始记录,可根据实际情况调整。为啥需要这个配置?有3点原因如下:
21    -   LOS\_MemAlloc接口内部也有函数调用;
22    -   外部可能对LOS\_MemAlloc接口有封装;
23    -   LOSCFG\_MEM\_RECORD\_LR\_CNT 配置的LR层数有限;
24
25
26正确配置这个宏,将无效的LR层数忽略,就可以记录有效的LR层数,节省内存消耗。
27
28## 开发指导<a name="section95828159308"></a>
29
30### 开发流程<a name="section369844416304"></a>
31
32该调测功能可以分析关键的代码逻辑中是否存在内存泄漏。开启这个功能,每次申请内存时,会记录LR信息。在需要检测的代码段前后,调用LOS\_MemUsedNodeShow接口,每次都会打印指定内存池已使用的全部节点信息,对比前后两次的节点信息,新增的节点信息就是疑似泄漏的内存节点。通过LR,可以找到具体申请的代码位置,进一步确认是否泄漏。
33
34调用LOS\_MemUsedNodeShow接口输出的节点信息格式如下:每1行为一个节点信息;第1列为节点地址,可以根据这个地址,使用GDB等手段查看节点完整信息;第2列为节点的大小,等于节点头大小+数据域大小;第3\~5列为函数调用关系LR地址,可以根据这个值,结合汇编文件,查看该节点具体申请的位置。
35
36```
37node        size   LR[0]      LR[1]       LR[2]
380x10017320: 0x528 0x9b004eba  0x9b004f60  0x9b005002
390x10017848: 0xe0  0x9b02c24e  0x9b02c246  0x9b008ef0
400x10017928: 0x50  0x9b008ed0  0x9b068902  0x9b0687c4
410x10017978: 0x24  0x9b008ed0  0x9b068924  0x9b0687c4
420x1001799c: 0x30  0x9b02c24e  0x9b02c246  0x9b008ef0
430x100179cc: 0x5c  0x9b02c24e  0x9b02c246  0x9b008ef0
44```
45
46>![](../public_sys-resources/icon-caution.gif) **注意:**
47>开启内存检测会影响内存申请的性能,且每个内存节点都会记录LR地址,内存开销也加大。
48
49### 编程实例<a name="section460801313313"></a>
50
51本实例实现如下功能:构建内存泄漏代码段。
52
531.  调用LOS\_MemUsedNodeShow接口,输出全部节点信息打印;
542.  申请内存,但没有释放,模拟内存泄漏;
553.  再次调用LOS\_MemUsedNodeShow接口,输出全部节点信息打印;
564.  将两次log进行对比,得出泄漏的节点信息;
575.  通过LR地址,找出泄漏的代码位置;
58
59### 示例代码<a name="section96539275311"></a>
60
61代码实现如下:
62
63```
64#include <stdio.h>
65#include <string.h>
66#include "los_memory.h"
67#include "los_config.h"
68
69void MemLeakTest(void)
70{
71    LOS_MemUsedNodeShow(LOSCFG_SYS_HEAP_ADDR);
72    void *ptr1 = LOS_MemAlloc(LOSCFG_SYS_HEAP_ADDR, 8);
73    void *ptr2 = LOS_MemAlloc(LOSCFG_SYS_HEAP_ADDR, 8);
74    LOS_MemUsedNodeShow(LOSCFG_SYS_HEAP_ADDR);
75}
76```
77
78### 结果验证<a name="section20527343183119"></a>
79
80编译运行输出log如下:
81
82```
83node         size   LR[0]       LR[1]       LR[2]
840x20001b04:  0x24   0x08001a10  0x080035ce  0x080028fc
850x20002058:  0x40   0x08002fe8  0x08003626  0x080028fc
860x200022ac:  0x40   0x08000e0c  0x08000e56  0x0800359e
870x20002594:  0x120  0x08000e0c  0x08000e56  0x08000c8a
880x20002aac:  0x56   0x08000e0c  0x08000e56  0x08004220
89
90node         size   LR[0]       LR[1]       LR[2]
910x20001b04:  0x24   0x08001a10  0x080035ce  0x080028fc
920x20002058:  0x40   0x08002fe8  0x08003626  0x080028fc
930x200022ac:  0x40   0x08000e0c  0x08000e56  0x0800359e
940x20002594:  0x120  0x08000e0c  0x08000e56  0x08000c8a
950x20002aac:  0x56   0x08000e0c  0x08000e56  0x08004220
960x20003ac4:  0x1d   0x08001458  0x080014e0  0x080041e6
970x20003ae0:  0x1d   0x080041ee  0x08000cc2  0x00000000
98```
99
100对比两次log,差异如下,这些内存节点就是疑似泄漏的内存块:
101
102```
1030x20003ac4:  0x1d   0x08001458  0x080014e0  0x080041e6
1040x20003ae0:  0x1d   0x080041ee  0x08000cc2  0x00000000
105```
106
107部分汇编文件如下:
108
109```
110                MemLeakTest:
111  0x80041d4: 0xb510         PUSH     {R4, LR}
112  0x80041d6: 0x4ca8         LDR.N    R4, [PC, #0x2a0]       ; g_memStart
113  0x80041d8: 0x0020         MOVS     R0, R4
114  0x80041da: 0xf7fd 0xf93e  BL       LOS_MemUsedNodeShow    ; 0x800145a
115  0x80041de: 0x2108         MOVS     R1, #8
116  0x80041e0: 0x0020         MOVS     R0, R4
117  0x80041e2: 0xf7fd 0xfbd9  BL       LOS_MemAlloc           ; 0x8001998
118  0x80041e6: 0x2108         MOVS     R1, #8
119  0x80041e8: 0x0020         MOVS     R0, R4
120  0x80041ea: 0xf7fd 0xfbd5  BL       LOS_MemAlloc           ; 0x8001998
121  0x80041ee: 0x0020         MOVS     R0, R4
122  0x80041f0: 0xf7fd 0xf933  BL       LOS_MemUsedNodeShow    ; 0x800145a
123  0x80041f4: 0xbd10         POP      {R4, PC}
124  0x80041f6: 0x0000         MOVS     R0, R0
125```
126
127其中,通过查找0x080041ee,就可以发现该内存节点是在MemLeakTest接口里申请的且是没有释放的。
128
129