kernel-small-debug-memory-info.md 3.8 KB
Newer Older
D
duangavin123 已提交
1
# 内存信息统计
D
duangavin123 已提交
2

D
duangavin123 已提交
3 4 5 6 7
- [基础概念](#基础概念)
- [功能配置](#功能配置)
- [开发指导](#开发指导)
  - [开发流程](#开发流程)
  - [编程实例](#编程实例)
D
duangavin123 已提交
8

D
duangavin123 已提交
9
## 基础概念
D
duangavin123 已提交
10 11 12

内存信息包括内存池大小、内存使用量、剩余内存大小、最大空闲内存、内存水线、内存节点数统计、碎片率等。

D
duangavin123 已提交
13 14 15 16 17 18
- 内存水线:即内存池的最大使用量,每次申请和释放时,都会更新水线值,实际业务可根据该值,优化内存池大小;

- 碎片率:衡量内存池的碎片化程度,碎片率高表现为内存池剩余内存很多,但是最大空闲内存块很小,可以用公式(fragment=100-100\*最大空闲内存块大小/剩余内存大小)来度量

- 其他统计信息:调用接口LOS_MemInfoGet时,会扫描内存池的节点信息,统计出相关信息。

D
duangavin123 已提交
19

D
duangavin123 已提交
20
## 功能配置
D
duangavin123 已提交
21

D
duangavin123 已提交
22
LOSCFG_MEM_WATERLINE:开关宏,默认关闭;若需要打开这个功能,可以在配置项中开启“Debug-> Enable memory pool waterline or not”。如需获取内存水线,需要打开该配置。
D
duangavin123 已提交
23 24


D
duangavin123 已提交
25
## 开发指导
D
duangavin123 已提交
26 27


D
duangavin123 已提交
28
### 开发流程
D
duangavin123 已提交
29 30 31 32 33 34 35 36 37 38

关键结构体介绍:

```
typedef struct {
    UINT32 totalUsedSize;       // 内存池的内存使用量
    UINT32 totalFreeSize;       // 内存池的剩余内存大小
    UINT32 maxFreeNodeSize;     // 内存池的最大空闲内存块大小
    UINT32 usedNodeNum;         // 内存池的非空闲内存块个数
    UINT32 freeNodeNum;         // 内存池的空闲内存块个数
D
duangavin123 已提交
39
#if (LOSCFG_MEM_WATERLINE == 1)     // 默认关闭,可以通过menuconfig配置工具打开
D
duangavin123 已提交
40 41 42 43 44
    UINT32 usageWaterLine;      // 内存池的水线值
#endif
} LOS_MEM_POOL_STATUS;
```

D
duangavin123 已提交
45
- 内存水线获取:调用LOS_MemInfoGet接口,第1个参数是内存池首地址,第2个参数是LOS_MEM_POOL_STATUS类型的句柄,其中字段usageWaterLine即水线值。
D
duangavin123 已提交
46

D
duangavin123 已提交
47
- 内存碎片率计算:同样调用LOS_MemInfoGet接口,可以获取内存池的剩余内存大小和最大空闲内存块大小,然后根据公式(fragment=100-100\*最大空闲内存块大小/剩余内存大小)得出此时的动态内存池碎片率。
D
duangavin123 已提交
48

D
duangavin123 已提交
49 50

### 编程实例
D
duangavin123 已提交
51 52 53

本实例实现如下功能:

D
duangavin123 已提交
54 55 56 57 58 59
1. 创建一个监控任务,用于获取内存池的信息;

2. 调用LOS_MemInfoGet接口,获取内存池的基础信息;

3. 利用公式算出使用率及碎片率。

D
duangavin123 已提交
60

D
duangavin123 已提交
61
**示例代码**
D
duangavin123 已提交
62 63


D
duangavin123 已提交
64
代码实现如下:
D
duangavin123 已提交
65 66 67 68 69 70 71 72 73 74
```
#include <stdio.h>
#include <string.h>
#include "los_task.h"
#include "los_memory.h"
#include "los_config.h"

void MemInfoTaskFunc(void)
{
    LOS_MEM_POOL_STATUS poolStatus = {0};
D
duangavin123 已提交
75

D
duangavin123 已提交
76
  /* pool为要统计信息的内存地址,此处以OS_SYS_MEM_ADDR为例 */
D
duangavin123 已提交
77
    void *pool = OS_SYS_MEM_ADDR;
D
duangavin123 已提交
78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104
    LOS_MemInfoGet(pool, &poolStatus);
    /* 算出内存池当前的碎片率百分比 */
    unsigned char fragment = 100 - poolStatus.maxFreeNodeSize * 100 / poolStatus.totalFreeSize;
    /* 算出内存池当前的使用率百分比 */
    unsigned char usage = LOS_MemTotalUsedGet(pool) * 100 / LOS_MemPoolSizeGet(pool);
    printf("usage = %d, fragment = %d, maxFreeSize = %d, totalFreeSize = %d, waterLine = %d\n", usage, fragment, poolStatus.maxFreeNodeSize, 
           poolStatus.totalFreeSize, poolStatus.usageWaterLine);
}

int MemTest(void)
{
    unsigned int ret;
    unsigned int taskID;
    TSK_INIT_PARAM_S taskStatus = {0};
    taskStatus.pfnTaskEntry = (TSK_ENTRY_FUNC)MemInfoTaskFunc;
    taskStatus.uwStackSize  = 0x1000;
    taskStatus.pcName       = "memInfo";
    taskStatus.usTaskPrio   = 10;
    ret = LOS_TaskCreate(&taskID, &taskStatus);
    if (ret != LOS_OK) {
        printf("task create failed\n");
        return -1;
    }
    return 0;
}
```

D
duangavin123 已提交
105

D
duangavin123 已提交
106
**结果验证**
D
duangavin123 已提交
107

D
duangavin123 已提交
108

D
duangavin123 已提交
109 110
编译运行输出的结果如下:

D
duangavin123 已提交
111

D
duangavin123 已提交
112 113 114
```
usage = 22, fragment = 3, maxFreeSize = 49056, totalFreeSize = 50132, waterLine = 1414
```