当前位置: 首页 > news >正文

lmk内存压力测试工具mem-pressure源码剖析

背景:

android系统开发过程中,经常会遇到一些low memory kill的问题,在分析这些系统低内存导致被杀问题时候,经常因为不好复现而成为一个比较烦恼的阻碍。因为这种低内存问题本身就不属于一种功能操作类型的问题,属于一种系统当前可使用的内存少导致的问题,所以分析这类lmk低内存被杀的情况迫切需要一种可以帮助我们复现系统低内存的工具,今天马哥就给大家介绍一个内存压力工具mem-pressure详细使用和源码剖析。

mem-pressure工具实战展示

系统的mem-pressure并不是自带集成的,需要集成这个mem-pressure工具是需要自己进行额外编译的,所以要先进行编译mem-pressure的bin文件后再使用。
编译mem-pressure

test@test:~/aosp15$ make mem-pressure
============================================
PLATFORM_VERSION_CODENAME=VanillaIceCream
PLATFORM_VERSION=VanillaIceCream
TARGET_PRODUCT=sdk_phone64_x86_64
TARGET_BUILD_VARIANT=eng
TARGET_ARCH=x86_64
TARGET_ARCH_VARIANT=x86_64
TARGET_2ND_ARCH_VARIANT=x86_64
HOST_OS=linux
HOST_OS_EXTRA=Linux-5.15.0-130-generic-x86_64-Ubuntu-20.04.3-LTS
HOST_CROSS_OS=windows
BUILD_ID=AP3A.241005.015.A2
OUT_DIR=out
============================================
[100% 6/6 1s remaining] Install: out/target/product/emu64x/system/bin/mem-pressure#### build completed successfully (6 seconds) ####

注意这里可以看到可以成功编译出mem-pressure

然后在push到手机设备上

test@test:~/aosp15$ adb push out/target/product/emu64x/system/bin/mem-pressure /data/local/tmp/
out/target/product/emu64x/system/bin/mem-pressure: 1 file pushed, 0 skipped. 82.2 MB/s (51264 bytes in 0.001s)

接下来既可以正常使用mem-pressure

130|emu64x:/data/local/tmp # ./mem-pressure -h                                                                                                                                                                    
Usage: [OPTIONS]-d N: Duration in microsecond to sleep between each allocation.-i N: Number of iterations to run the alloc process.-o N: The oom_score to set the child process to before alloc.-s N: Number of bytes to allocate in an alloc process loop.
Aborted 

可以看到mem-pressure可以设置一些参数
正常使用也可以不带任何的参数,都使用默认的:

emu64x:/data/local/tmp # ./mem-pressure                                                                                                                                                                           
Child 0 allocated 5936 MB
Child 1 allocated 5968 MB

可以到执行mem-pressure后有多个子进程在申请5936 MB内存,同时看看logcat是否有lowmemorykiller相关打印:

在这里插入图片描述

源码分析

源码路径:
system/extras/alloc-stress/mem-pressure.cpp
完整源码:
先看main方法

void usage() {printf("Usage: [OPTIONS]\n\n""  -d N: Duration in microsecond to sleep between each allocation.\n""  -i N: Number of iterations to run the alloc process.\n""  -o N: The oom_score to set the child process to before alloc.\n""  -s N: Number of bytes to allocate in an alloc process loop.\n");
}int main(int argc, char* argv[]) {pid_t pid;size_t* shared;int c, i = 0;size_t duration = 1000;int iterations = 0;const char* oom_score = "899";size_t step_size = 2 * 1024 * 1024;  // 2 MBsize_t size = step_size;while ((c = getopt(argc, argv, "hi:d:o:s:")) != -1) {switch (c) {case 'i':iterations = atoi(optarg);break;case 'd':duration = atoi(optarg);break;case 'o':oom_score = optarg;break;case 's':step_size = atoi(optarg);break;case 'h':usage();abort();default:abort();}}shared = (size_t*)mmap(NULL, sizeof(size_t), PROT_READ | PROT_WRITE, MAP_ANONYMOUS | MAP_SHARED,0, 0);while (iterations == 0 || i < iterations) {*shared = 0;pid = fork();//不断for新的子进程if (!pid) {/* Child */add_pressure(shared, size, step_size, duration, oom_score);//子进程开始添加内存压力,这个是核心方法/* Shoud not get here */exit(0);} else {wait(NULL);//主进程一直等待,等到子进程内存到达极限死了,才会进行往下执行printf("Child %d allocated %zd MB\n", i, *shared / 1024 / 1024);//打印出子进程已经申请了多少内存size = *shared / 2;}i++;}
}

可以总结一下mem-pressure核心如下:
1、不断创建新的子进程
2、每个子进程进行相关的内存申请
3、子进程内存达到极限被杀了后会打印出申请的内存值

再看具体是怎么给进程内存加压的

//申请指定内存大小并进行设置为0
void* alloc_set(size_t size) {void* addr = NULL;addr = malloc(size);if (!addr) {printf("Allocating %zd MB failed\n", size / 1024 / 1024);} else {memset(addr, 0, size);}return addr;
}void add_pressure(size_t* shared, size_t size, size_t step_size, size_t duration,const char* oom_score) {int fd, ret;fd = open("/proc/self/oom_score_adj", O_WRONLY);//给进程的oom_score_adj写入相关adj值的文件ret = write(fd, oom_score, strlen(oom_score));//写人对应的oom_score到上面文件if (ret < 0) {printf("Writing oom_score_adj failed with err %s\n", strerror(errno));}close(fd);if (alloc_set(size)) {*shared = size;}
//这里会一直申请内存while (alloc_set(step_size)) {size += step_size;*shared = size;usleep(duration);//这里会有个延时1000us}
}

更多framework技术干货,请关注下面“千里马学框架”

http://www.lryc.cn/news/531779.html

相关文章:

  • 企业四要素如何用Java进行调用
  • 修剪二叉搜索树(力扣669)
  • 一款由 .NET 官方团队开源的电子商务系统 - eShop
  • 论最新技术编程类有什么,值得关注的点有什么呢?
  • Java入门进阶
  • Java并发编程面试题:ThreadLocal(8题)
  • Zabbix7.0安装(Ubuntu24.04+LNMP)
  • 从 0 到 1 构建数仓之DWD层
  • S4 HANA手工记账Tax Payable – FB41
  • 【自然语言处理(NLP)】NLP实战:IMDB影评情感分析项目
  • DIY Shell:探秘进程构建与命令解析的核心原理
  • 通过Redisson构建延时队列并实现注解式消费
  • SQL Server配置管理器无法连接到 WMI 提供程序
  • Linux内核源码:ext4 extent详解
  • Maven jar 包下载失败问题处理
  • 自指学习:AGI的元认知突破
  • 排序算法--希尔排序
  • Java 2024年面试总结(持续更新)
  • TensorFlow是个啥玩意?
  • 不可信的搜索路径(CWE-426)
  • Linux——基础命令
  • 利用TensorFlow.js实现浏览器端机器学习:一个全面指南
  • 利用HTML和css技术编写学校官网页面
  • SpringSecurity密码编码器:使用BCrypt算法加密、自定义密码编码器
  • 笔记:新能源汽车零部件功率级测试怎么进行?
  • ES6中的map和原生的对象有什么区别?
  • 2502vim,vim文本对象中文文档
  • spring security与gateway结合进行网关鉴权和授权
  • LabVIEW在电机自动化生产线中的实时数据采集与生产过程监控
  • log4j2日志配置文件