bootchart 使用说明及代码分析--android启动优化

2023-10-24 23:50

本文主要是介绍bootchart 使用说明及代码分析--android启动优化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

bootchart是一个用于linux启动过程性能分析的开源软件工具,在系统启动过程自动收集CPU占用率、进程等信息,并以图形方式显示分析结果,可用作指导优化系统启动过程。


bootchart是一个对linux启动流程进行分析得开源软件工具。android中有集成bootchart源码,路径为system/core/init/bootchart.c

 
第一部分:先从具体使用流程如下


1、编译android中的bootchart(缺省时不被编译)
  在android源码system/core/init/目录执行: mm INIT_BOOTCHART=true -B

  或者直接修改 Android.mak文件

[html] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. <SPAN style="COLOR: #333333; FONT-SIZE: 14px">  LOCAL_PATH:= $(call my-dir)  
  2.     include $(CLEAR_VARS)  
  3.       
  4.     INIT_BOOTCHART :true  </SPAN>  
<span style="font-size:14px;color:#333333;">	LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)
INIT_BOOTCHART := true  </span>

2、将新编译的android系统镜像烧录到android设备中。
编译生成新的可执行文件init,该文件在手机文件系统位于根/下,对应的flash image是boot.img,为此需重新烧写含有新的init的boot.img


3、在手机上创建文件/data/bootchart-start,其内容是bootchart的采样时间
adb shell 'echo $TIMEOUT > /data/bootchart-start'
其中$TIMEOUT是期望采样的时间,单位为秒,例如要采样两分钟,则执行:
adb shell 'echo 120 > /data/bootchart-start'


4、重启设备,init运行时将自动创建文件夹/data/bootchart/,并在其中保存采样数据,采样数据由5个文件组成:

[html] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. <SPAN style="COLOR: #333333; FONT-SIZE: 14px">-rw-rw-rw- root     root          732 1970-01-01 08:00 header  
  2. -rw-r--r-- root     root            0 1970-01-01 08:00 kernel_pacct  
  3. -rwxr-xr-x root     root       517150 2014-04-09 12:06 proc_diskstats.log  
  4. -rwxr-xr-x root     root      2783967 2014-04-09 12:06 proc_ps.log  
  5. -rwxr-xr-x root     root       152090 2014-04-09 12:06 proc_stat.log</SPAN>  
<span style="font-size:14px;color:#333333;">-rw-rw-rw- root     root          732 1970-01-01 08:00 header
-rw-r--r-- root     root            0 1970-01-01 08:00 kernel_pacct
-rwxr-xr-x root     root       517150 2014-04-09 12:06 proc_diskstats.log
-rwxr-xr-x root     root      2783967 2014-04-09 12:06 proc_ps.log
-rwxr-xr-x root     root       152090 2014-04-09 12:06 proc_stat.log</span>

需要注意,在手机上运行bootchart采样完成后若不再使用bootchart则需手工删除文件/data/bootchart-start,否则手机每次重启时都会运行bootchart。


5、文件打包
在/data/bootchart/目录下执行命令:


[html] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. <SPAN style="COLOR: #333333; FONT-SIZE: 14px">busybox tar -czf bootchart.tgz header proc_stat.log proc_ps.log proc_diskstats.log kernel_pacct </SPAN>  
<span style="font-size:14px;color:#333333;">busybox tar -czf bootchart.tgz header proc_stat.log proc_ps.log proc_diskstats.log kernel_pacct </span>

然后将生成bootchart.tgz用u盘拷贝到电脑(ubuntu系统)上。

也可手工通过adb pull将这5个文件上传到PC并打包,反正最终就是生成 bootchart.tgz 


6.在电脑上安装bootchart工具

使用sudo apt-get install bootchart 命令安装, 如果出现 bootchart无法正常解析android中生成的bootchart.tgz文件。
需要使用老版本的安装包bootchart_0.9-0ubuntu6_all.deb,可以在此下载http://download.csdn.net/detail/sckgenius/7166477。
先sudo apt-get install librsvg2-bin,然后sudo dpkg -i bootchart_0.9-0ubuntu6_all.deb 。


7、执行下面的命令生成分析结果图表,缺省生成png格式的图像文件bootchart.png:
java -jar /usr/share/bootchart/bootchart.jar /path/to/bootchart.tgz

这里上传一张bootchart.png图:



第二部分:下面说一下bootchart是如何得到这些数据信息的


分析是源码:system\core\init\bootchart.c 只有一个文件


1、启动

init.c 中通过宏定义 BOOTCHART 增加代码

[cpp] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. <SPAN style="COLOR: #333333; FONT-SIZE: 14px">#if BOOTCHART  
  2. static int bootchart_init_action(int nargs, char **args)  
  3. {  
  4.     bootchart_count = bootchart_init();  
  5.     if (bootchart_count < 0) {  
  6.         ERROR("bootcharting init failure\n");  
  7.     } else if (bootchart_count > 0) {  
  8.         NOTICE("bootcharting started (period=%d ms)\n", bootchart_count*BOOTCHART_POLLING_MS);  
  9.     } else {  
  10.         NOTICE("bootcharting ignored\n");  
  11.     }  
  12.   
  13.     return 0;  
  14. }  
  15. #endif</SPAN>  
<span style="font-size:14px;color:#333333;">#if BOOTCHART
static int bootchart_init_action(int nargs, char **args)
{
bootchart_count = bootchart_init();
if (bootchart_count < 0) {
ERROR("bootcharting init failure\n");
} else if (bootchart_count > 0) {
NOTICE("bootcharting started (period=%d ms)\n", bootchart_count*BOOTCHART_POLLING_MS);
} else {
NOTICE("bootcharting ignored\n");
}
return 0;
}
#endif</span>

通过调用 bootchart_init() 启动


2、周期性执行

int main(int argc, char **argv)

[cpp] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. <SPAN style="FONT-SIZE: 14px"><SPAN style="COLOR: #333333">{  
  2.   
  3. #if BOOTCHART  
  4.     queue_builtin_action(bootchart_init_action, "bootchart_init");  
  5. #endif  
  6.   
  7.     for(;;) {  
  8.         int nr, i, timeout = -1;  
  9.   
  10.   
  11. #if BOOTCHART  
  12.         if (bootchart_count > 0) {  
  13.             if (timeout < 0 || timeout > BOOTCHART_POLLING_MS)  
  14.                 timeout = BOOTCHART_POLLING_MS;  
  15.             if (</SPAN><STRONG><SPAN style="COLOR: #ff0000">bootchart_step(</SPAN><SPAN style="COLOR: #cc0000">)</SPAN></STRONG><SPAN style="COLOR: #333333"> < 0 || --bootchart_count == 0) {  
  16.                 bootchart_finish();  
  17.                 bootchart_count = 0;  
  18.             }  
  19.         }  
  20. #endif   
  21.   
  22.         nr = poll(ufds, fd_count, timeout);  
  23.           
  24.   }   
  25.     ...  
  26. }</SPAN></SPAN>  
<span style="font-size:14px;"><span style="color:#333333;">{
#if BOOTCHART
queue_builtin_action(bootchart_init_action, "bootchart_init");
#endif
for(;;) {
int nr, i, timeout = -1;
#if BOOTCHART
if (bootchart_count > 0) {
if (timeout < 0 || timeout > BOOTCHART_POLLING_MS)
timeout = BOOTCHART_POLLING_MS;
if (</span><strong><span style="color:#ff0000;">bootchart_step(</span><span style="color:#cc0000;">)</span></strong><span style="color:#333333;"> < 0 || --bootchart_count == 0) {
bootchart_finish();
bootchart_count = 0;
}
}
#endif
nr = poll(ufds, fd_count, timeout);
}	
...
}</span></span>

默认周期时间:
# define BOOTCHART_POLLING_MS   200   /* polling period in ms */


3、具体如何采样数据

分析一下源码就很清楚了,就是通过linux中的标准命令(shell上执行cat xxx 类似)并将其结果写入相应的文件

/proc/cmdline

/proc/version
/proc/cpuinfo
写入到文件 #define LOG_HEADER      LOG_ROOT"/header"

/proc/uptime
/proc/stat
写入到文件 #define LOG_STAT        LOG_ROOT"/proc_stat.log"

/proc/uptime
/proc/diskstats
写入到文件 #define LOG_DISK        LOG_ROOT"/proc_diskstats.log"

/proc/uptime
/proc/$PID/cmdline
/proc/$PID/stat
写入到文件 #define LOG_PROCS       LOG_ROOT"/proc_ps.log"

这个文件只打开,没有写入什么内容
/* create kernel process accounting file */
#define LOG_ACCT        LOG_ROOT"/kernel_pacct"


参考资料:

1、system/core/init/README.BOOTCHART

2、http://www.bootchart.org/index.html

这篇关于bootchart 使用说明及代码分析--android启动优化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/278604

相关文章

vue使用docxtemplater导出word

《vue使用docxtemplater导出word》docxtemplater是一种邮件合并工具,以编程方式使用并处理条件、循环,并且可以扩展以插入任何内容,下面我们来看看如何使用docxtempl... 目录docxtemplatervue使用docxtemplater导出word安装常用语法 封装导出方

SpringBoot3实现Gzip压缩优化的技术指南

《SpringBoot3实现Gzip压缩优化的技术指南》随着Web应用的用户量和数据量增加,网络带宽和页面加载速度逐渐成为瓶颈,为了减少数据传输量,提高用户体验,我们可以使用Gzip压缩HTTP响应,... 目录1、简述2、配置2.1 添加依赖2.2 配置 Gzip 压缩3、服务端应用4、前端应用4.1 N

Linux换行符的使用方法详解

《Linux换行符的使用方法详解》本文介绍了Linux中常用的换行符LF及其在文件中的表示,展示了如何使用sed命令替换换行符,并列举了与换行符处理相关的Linux命令,通过代码讲解的非常详细,需要的... 目录简介检测文件中的换行符使用 cat -A 查看换行符使用 od -c 检查字符换行符格式转换将

Go标准库常见错误分析和解决办法

《Go标准库常见错误分析和解决办法》Go语言的标准库为开发者提供了丰富且高效的工具,涵盖了从网络编程到文件操作等各个方面,然而,标准库虽好,使用不当却可能适得其反,正所谓工欲善其事,必先利其器,本文将... 目录1. 使用了错误的time.Duration2. time.After导致的内存泄漏3. jsO

使用Jackson进行JSON生成与解析的新手指南

《使用Jackson进行JSON生成与解析的新手指南》这篇文章主要为大家详细介绍了如何使用Jackson进行JSON生成与解析处理,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 核心依赖2. 基础用法2.1 对象转 jsON(序列化)2.2 JSON 转对象(反序列化)3.

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

使用Python实现快速搭建本地HTTP服务器

《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

Elasticsearch 在 Java 中的使用教程

《Elasticsearch在Java中的使用教程》Elasticsearch是一个分布式搜索和分析引擎,基于ApacheLucene构建,能够实现实时数据的存储、搜索、和分析,它广泛应用于全文... 目录1. Elasticsearch 简介2. 环境准备2.1 安装 Elasticsearch2.2 J

使用C#代码在PDF文档中添加、删除和替换图片

《使用C#代码在PDF文档中添加、删除和替换图片》在当今数字化文档处理场景中,动态操作PDF文档中的图像已成为企业级应用开发的核心需求之一,本文将介绍如何在.NET平台使用C#代码在PDF文档中添加、... 目录引言用C#添加图片到PDF文档用C#删除PDF文档中的图片用C#替换PDF文档中的图片引言在当

Java中List的contains()方法的使用小结

《Java中List的contains()方法的使用小结》List的contains()方法用于检查列表中是否包含指定的元素,借助equals()方法进行判断,下面就来介绍Java中List的c... 目录详细展开1. 方法签名2. 工作原理3. 使用示例4. 注意事项总结结论:List 的 contain