Fork me on GitHub
侧边栏

android启动时间测试

需要对比基线测试的情况;

Kernel log: adb shell dmesg

Event log: adb shell logcat –b events

Logcat log: adb shell logcat

lk boottime测试:(只适合LE)

adb shell dmesg

搜索关键字:
[0.424253] KPI: Boot loader start count = 35880
[0.424265] KPI: Boot loader end count = 73583
[0.424271] KPI: Boot loader display count = 48572
[0.424277] KPI: Boot loader load kernel count = 3318
[0.424285] KPI: Kernel MPM timestamp = 94543
[0.424290] KPI: Kernel MPM clock frequency = 32768

Boot loader time calculation

一般要使用GKI 的编译,否则这个计时可能不准确。

  • Hw_count = 94543

  • Frequency = 32768

  • Kernel timestamp = 0.424285

  • Boot loader time = (Hw_count/Frequency) – Kernel timestamp

  • Boot loader time = (94543/32768) – 0.424285 = 2.460938388671875 sec

Total boot up time = Boot loader time + HLOS boot up time (from boot_progress_enable_screen)

kernel阶段的log:(demsg.log)

The following are the kernel Init stages:-

  • First stage: Kernel driver initialization takes place

  • Second stage: File system mount takes place`
    grep "init:"
    [ 0.954087] init: init first stage started!
    [ 8.398016] init: init second stage started!

init阶段的log:(logcat_event.log)

搜索关键字:boot_progress


boot_progress|bootAnimation:|wm_boot_animation_done

boot_progress_start: 6914   标志着kernel启动完成。Zygote开始启动  此时界面还是现实开机静态图片
 BootAnimation: BootAnimationStartTiming start time: 15202ms    界面开始开机动画
boot_progress_preload_start: 9509  Zygote开始加载资源
boot_progress_preload_end: 12251 Zygote加载资源结束
boot_progress_system_run: 12619 SystemServer开始启动
boot_progress_pms_start: 13424 PMS开始启动
boot_progress_pms_system_scan_start: 13482 PMS扫描/system目录下的安装包
boot_progress_pms_data_scan_start: 15598 PMS扫描/data目录下的安装包
boot_progress_pms_scan_end: 15633 PMS扫描结束
boot_progress_pms_ready: 16225 PMS初始化完毕
boot_progress_ams_ready: 22036 AMS就绪
android.intent.category.HOME: 先启动“Android正在启动” FallbackHome。
boot_progress_enable_screen: 24814 AMS启动完成后开始激活屏幕,从此以后

注意这里的boot_progress不包含bootloader的时间
boot_progress_enable_screen代表完整的开机日志。
看完event.log之后,可以大致推算出耗时点,如果是SystemServer耗时,可以在android.log里面看每个服务的启动时间推算卡在哪个服务,搜索关键字“SystemServer”即可。

service-start stage

logcat中搜索关键字:init:
[ 0.954087] init: init first stage started!
[ 0.975432] init: Loading module /lib/modules/qcom-cpufreq-hw.ko with args ''
[ 1.033704] init: Loaded kernel module /lib/modules/qcom-cpufreq-hw.ko
[ 1.033866] init: Loading module /lib/modules/sched-walt.ko with args ''
[ 1.050295] init: Loaded kernel module /lib/modules/sched-walt.ko
[ 1.051146] init: Loading module /lib/modules/minidump.ko with args ''
[ 1.056404] init: Loaded kernel module /lib/modules/minidump.ko
[ 1.056556] init: Loading module /lib/modules/msm_rtb.ko with args 'filter=0x237 filter=0x237'
[ 1.065098] init: Loaded kernel module /lib/modules/msm_rtb.ko
[ 1.065552] init: Loading module /lib/modules/qcom_ipc_logging.ko with args ''
[ 5.581554] sdhci_msm 4744000.sdhci: mmc0: CQE init: success
[ 6.283528] init: [libfs_mgr] Created logical partition scratch on device /dev/block/dm-6
[ 7.091009] init: [libfs_mgr] __mount(source=/dev/block/dm-6,target=/mnt/scratch,type=f2fs)=0: Success
[ 7.156231] init: [libfs_mgr] __mount(source=/dev/block/dm-6,target=/mnt/scratch,type=f2fs)=0: Success
[ 7.171745] init: [libfs_mgr] __mount(target=/system,flag=MS_PRIVATE)=-1: Invalid argument
[ 7.193904] init: [libfs_mgr] __mount(source=overlay,target=/system,type=overlay,upperdir=/mnt/scratch/overlay/system/upper)=0
[ 7.223198] init: [libfs_mgr] __mount(source=overlay,target=/system_ext,type=overlay,upperdir=/mnt/scratch/overlay/system_ext/upper)=0
[ 7.252759] init: [libfs_mgr] __mount(source=overlay,target=/product,type=overlay,upperdir=/mnt/scratch/overlay/product/upper)=0
[ 7.281241] init: [libfs_mgr] __mount(source=overlay,target=/vendor,type=overlay,upperdir=/mnt/scratch/overlay/vendor/upper)=0
[ 7.309920] init: [libfs_mgr] __mount(source=overlay,target=/vendor_dlkm,type=overlay,upperdir=/mnt/scratch/overlay/vendor_dlkm/upper)=0
[ 7.340022] init: [libfs_mgr] __mount(source=overlay,target=/system_dlkm,type=overlay,upperdir=/mnt/scratch/overlay/system_dlkm/upper)=0
[ 7.413005] printk: init: 181 output lines suppressed due to ratelimiting
[ 7.587287] init: [libfstab] Using Android DT directory /proc/device-tree/firmware/android/
[ 7.597363] init: [libfstab] dt_fstab: Skip disabled entry for partition vendor
[ 7.605818] init: [libfstab] dt_fstab: Skip disabled entry for partition system
[ 7.625867] init: Opening SELinux policy
[ 7.652166] init: Loading SELinux policy
[ 8.398016] init: init second stage started!
[ 8.669245] init: [libfstab] Using Android DT directory /proc/device-tree/firmware/android/
[ 8.705971] init: Overriding previous property 'ro.logd.size':'1M' with new value '1048576'
[ 8.716936] audit: type=1107 audit(484.615:7): pid=1 uid=0 auid=4294967295 ses=4294967295 subj=u:r:init:s0 msg='avc: denied { set } for property=persist.backup.ntpServer pid=1 uid=0 gid=0 scontext=u:r:vendor_init:s0 tcontext=u:object_r:default_prop:s0 tclass=property_service permissive=1'
[ 8.717393] init: Overriding previous property 'dalvik.vm.heapstartsize':'32m' with new value '8m'
[ 8.754074] init: Overriding previous property 'dalvik.vm.heapsize':'256m' with new value '128m'

使用 bootchart 分析开机启动时间

bootchart 是一个能对 GNU/Linux boot 过程进行性能分析并把结果直观化的开源工具,在系统启动过程中自动收集 CPU 占用率、磁盘吞吐率、进程等信息,并以图形方式显示分析结果,可用作指导优化系统启动过程。BootChart 包含数据收集工具和图像产生工具,数据收集工具在原始的 BootChart 中是独立的 shell 程序,但在 Android 中,数据收集工具被集成到了 init 程序中。

在 Android 中会通过一个叫 do_bootchart_start 的函数来判断是否抓取 bootchar 数据

// system/core/init/bootchart.cpp
static Result<Success> do_bootchart_start() {
    // We don't care about the content, but we do care that /data/bootchart/enabled actually exists.
    std::string start;
     // 只要存在/data/bootchart/enabled文件,即抓取 bootchart 数据
    if (!android::base::ReadFileToString("/data/bootchart/enabled", &start)) {
        LOG(VERBOSE) << "Not bootcharting";
        return Success();
    }

    g_bootcharting_thread = new std::thread(bootchart_thread_main);
    return Success();
}

所以我们只要创建一个 /data/bootchart/enabled 即可开启 bootchar 数据的抓取。

system/core/init/README.md

android 12: https://android-review.googlesource.com/c/platform/system/sepolicy/+/1888457

adb shell touch /data/bootchart/enabled
adb reboot

接着我们准备用于生成 bootchart 图的分析软件 pybootchartgui:

# 下载 pybootchartgui
sudo apt install python-is-python3
cd ~/Documents
git clone https://github.com/xrmx/bootchart.git
cd bootchart/pybootchartgui
mv main.py.in main.py
# 建立
sudo ln -s ~/Project/bootchart-master/pybootchartgui.py /usr/bin/pybootchartgui

回到 Android 源码目录下执行:

system/core/init/grab-bootchart.sh
parsing '/tmp/android-bootchart/bootchart.tgz'
parsing 'header'
parsing 'proc_stat.log'
parsing 'proc_ps.log'
parsing 'proc_diskstats.log'
merged 0 logger processes
pruned 47 process, 0 exploders, 2 threads, and 1 runs
bootchart written to 'bootchart.png'
Clean up /tmp/android-bootchart/ and ./bootchart.png when done

在源码目录下就生成了 bootchart.png 图:

image

从生成的图片可以更加直观详细的看到开机耗时以及硬件使用情况.

抓取 trace 文件分析开机启动时间

修改 frameworks/native/cmds/atrace/atrace.rc 中 service boottrace 的内容:

service boottrace /system/bin/atrace --async_start -b 30720 gfx input view webview wm am sm audio video binder_lock binder_driver camera hal res dalvik rs bionic power pm ss database network adb vibrator aidl sched  
    disabled
    oneshot

接着重新编译源码,启动虚拟机。

打开抓取 boottrace 的属性开关

adb  shell setprop persist.debug.atrace.boottrace 0

生成和拉取 boottrace 文件

adb shell atrace --async_stop -z -c -o /data/local/tmp/boot_trace
adb  pull /data/local/tmp/boot_trace

自此,我们就得到了启动阶段的 trace 文件,可以通过 perfetto 工具打开并分析它了。

开机启动优化

开启启动优化可以分为两类:

  • 程序异常,导致开机时间加长或无法开机
  • 正常开机时间的基础上加快开机速度

第一种情况,很多时候是某个模块的魔改导致的,我们的主要任务是找到这个模块,一般通过上一节介绍的三种开机时间分析的手段就可以很容易的找到了。

更多的时候是第二种情况,Android 启动可以优化的阶段主要有:

  • Bootloader
  • Kernel
  • Framework

我们主要关注 Framework 阶段的优化,常见的优化手段有:

在 Zygote 进程启动后,加载自定义的驱动,启动自定义的进程

zygote 是在 late-init 阶段启动

on late-init
    trigger early-fs
    trigger fs
    trigger post-fs
    trigger late-fs
    trigger post-fs-data
    trigger load_persist_props_action
    # zygote 启动
    trigger zygote-start
    trigger firmware_mounts_complete
    trigger early-boot
    trigger boot

加载自定义的驱动,启动自定义的进程两类操作,我们可以放到 early-boot 或者 boot 阶段启动,这样可以让 Zygote 进程今早的启动,加快系统的启动。

on boot
    insmod /vendor/lib/modules/btusb.ko
    start xxxx
    # .....

调整 Zygote 启动参数

service zygote /system/bin/app_process -Xzygote /system/bin --zygote --start-system-server --enable-lazy-preload
    class main
    priority -20
    user root
    group root readproc reserved_disk
    socket zygote stream 660 root system
    socket usap_pool_primary stream 660 root system
    onrestart exec_background - system system -- /system/bin/vdc volume abort_fuse
    onrestart write /sys/power/state on
    onrestart restart audioserver
    onrestart restart cameraserver
    onrestart restart media
    onrestart restart netd
    onrestart restart wificond
    task_profiles ProcessCapacityHigh MaxPerformance

这里改动了两个地方 --enable-lazy-preloadtask_profiles ProcessCapacityHigh MaxPerformance

添加了 --enable-lazy-preload 参数,在开机阶段不会执行 preload 预加载操作。在开机后,通过 system-server 发送指令给 zygote 做资源加载操作,在发送指令前,system-server 会加载一部分自己使用的类,会和 zygote 中存在相同的一部分备份,所有会多耗费一些内存。

task_profiles ProcessCapacityHigh MaxPerformance 会让内核使用尽可能多的资源来启动 Zygote。

开机阶段 CPU 开启性能模式

# cpu 开启性能模式
on early-init
    write /sys/devices/system/cpu/cpu0/cpufreq/scaling_governor performance

# 开机完成后,CPU 频率变成自适应
on property:sys.boot_completed=1
    write /sys/devices/system/cpu/cpu0/cpufreq/scaling_governor schedutil

读写 IO 调整

可以通过调整事先预读数据的 Kb 数以及默认 IO 请求队列的长度来加快 IO 的读写以提高开机速度

# on late-fs
#     # boot time fs tune
#     write /sys/block/mmcblk0/queue/iostats 0
#     # 增大事先预读数据的 Kb 数
#     write /sys/block/mmcblk0/queue/read_ahead_kb 2048
#     # 默认 IO 请求队列的长度
#     write /sys/block/mmcblk0/queue/nr_requests 256

# on property:sys.boot_completed=1
#     # end boot time fs tune
#     write /sys/block/mmcblk0/queue/read_ahead_kb 128

on late-fs
  # boot time fs tune
    write /sys/block/sda/queue/iostats 0
    write /sys/block/sda/queue/scheduler cfq
    write /sys/block/sda/queue/iosched/slice_idle 0
    # 增大事先预读数据的 Kb 数
    write /sys/block/sda/queue/read_ahead_kb 2048
    # IO 请求队列的长度
    write /sys/block/sda/queue/nr_requests 256
    write /sys/block/dm-0/queue/read_ahead_kb 2048
    write /sys/block/dm-1/queue/read_ahead_kb 2048

on property:sys.boot_completed=1
    # end boot time fs tune
    write /sys/block/sda/queue/read_ahead_kb 512

移除没有用的模块

主要是修改 SystemServer,删除一些用不到的服务,比如有的产品是电视、音响,就不会用到电话,指纹、定位、打印等相关的模块。

一般可裁剪的模块有:

TelecomLoaderService
TelephonyRegistry
StatusBarManagerService
SearchManagerService
SerialService
FingerprintService
CameraService
MmsService

延时启动 persist app

修改 startPersistentApps 方法,延时启动 persist app

void startPersistentApps(int matchFlags) {
    if (mFactoryTest == FactoryTest.FACTORY_TEST_LOW_LEVEL) return;

    synchronized (this) {
        try {
            final List<ApplicationInfo> apps = AppGlobals.getPackageManager()
                    .getPersistentApplications(STOCK_PM_FLAGS | matchFlags).getList();
            for (ApplicationInfo app : apps) {
                if (!"android".equals(app.packageName)) {
                    mHandler.postDelayed(() -> {
                        addAppLocked(app, null, false, null /* ABI override */,
                                ZYGOTE_POLICY_FLAG_BATCH_LAUNCH);
                    }, 1000);
                }
            }
        } catch (RemoteException ex) {
        }
    }
}

精简 preload 的 classes

可以根据产品的类型修改 frameworks/base/config/preloaded-classes 文件,来删减一些用不到的 preload classes

常见可删除的预加载 classes 有:

// 生物识别
android.hardware.biometrics

// 人脸识别
android.hardware.face

// 打印服务
android.hardware.fingerprint
android.print.

// 部分定位相关, 还有GPS定位相关
android.hardware.location
com.android.internal.location.GpsNetInitiatedHandler
android.location.Gnss*

// 手机通话相关
android.telephony.
android.telecom.
com.android.i18n.phonenumbers.
com.android.ims
android.hardware.radio

// nfc相关
android.nfc.

如果需要再做一些大的裁剪,可以使用 frameworks\base\config\generate-preloaded-classes.sh 脚本来重新生成 preloaded-classes

参考文档

https://source.android.com/docs/core/architecture/kernel/boot-time-opt?hl=zh-cn
https://blog.csdn.net/gary_qing/article/details/134404800

posted @ 2024-07-30 15:00  yooooooo  阅读(641)  评论(0)    收藏  举报