1 简介

https://www.lizenghai.com/archives/26061.html

2 用法

用命令行启动Systrace抓取采样:《Capture a system trace on the command line

自定义采样的方法:《Define custom events》这个方法确实要仔细看,尤其是采样命令的描述,强调了要使用 -a 参数,之前我忽略了这一点,导致自定义的标签总也显示不出来。

python systrace.py -a com.autonavi.amapauto -b  -o my_systrace_report.html sched freq idle am wm gfx view binder_driver hal dalvik camera input res

【坑】采用Android Device Monitor来抓取Systrace时有一个坑,就是经常出现Java heap error(因为我自动插桩所有函数,导致Systrace标签数量庞大),后来发现原因在于Monitor配置的Java堆最大值太小了,收集trace数据时需要大量的堆(怀疑是做数据合并加工)引发堆空间错误。解决方法如下:

. 找到Monitor脚本所在目录,里面有lib\monitor-x86,例如:C:\Users\liuheng.klh\AppData\Local\Android\Sdk\tools\lib\monitor-x86(或monitor-x86_64)
. 打开monitor.ini 配置文件,修改如下三项空间配置(尽可能大点就行):
-XX:MaxPermSize=1024m
-Xms2048m
-Xmx4096m

(1)实现自定义标签

在Android 6.0(API 23)中包含trace.h,可以在Native层直接进行引用,获取函数指针进行调用。用 ATRACE_CALL 宏直接在需要插桩的函数入口调用一下即可。关键代码:

 #include <android/trace.h>

 /////////////////////////////////////////////////////////////////////
// utils.h
extern void *(*_ATrace_beginSection) (const char* sectionName) __attribute__ ((no_instrument_function));
extern void *(*_ATrace_endSection) (void) __attribute__ ((no_instrument_function));
extern bool *(*_ATrace_isEnabled) (void) __attribute__ ((no_instrument_function)); // SysTrace
class ScopedTrace {
const char* _name;
public:
static bool _bInit;
static void init() __attribute__ ((no_instrument_function)); typedef void *(*fp_ATrace_beginSection) (const char* sectionName);
typedef void *(*fp_ATrace_endSection) (void);
typedef bool *(*fp_ATrace_isEnabled) (void); public:
inline ScopedTrace(const char* name) __attribute__ ((no_instrument_function)) {
if (!_bInit) {
init();
}
_ATrace_beginSection(name) ; _name = name;
//LOGD("_ATrace_beginSection(%s) _ATrace_isEnabled()=%d", name, _ATrace_isEnabled());
} inline ~ScopedTrace() __attribute__ ((no_instrument_function)) {
_ATrace_endSection();
//LOGD("_ATrace_endSection(%s) _ATrace_isEnabled()=%d", _name, _ATrace_isEnabled());
}
}; #define ATRACE_NAME(name) ScopedTrace ___tracer(name)
#define ATRACE_CALL() ATRACE_NAME(__FUNCTION__) /////////////////////////////////////////////////////////////////////
// utils.cpp
void *(*_ATrace_beginSection) (const char* sectionName) = NULL;
void *(*_ATrace_endSection) (void) = NULL;
bool *(*_ATrace_isEnabled) (void) = NULL; bool ScopedTrace::_bInit = false;
void ScopedTrace::init() {
if (_bInit) {
return;
} // Retrieve a handle to libandroid.
void *lib = dlopen("libandroid.so", RTLD_NOW || RTLD_LOCAL); // Access the native tracing functions.
if (lib != NULL) {
// Use dlsym() to prevent crashes on devices running Android 5.1
// (API level 22) or lower.
_ATrace_beginSection = reinterpret_cast<fp_ATrace_beginSection>(
dlsym(lib, "ATrace_beginSection"));
_ATrace_endSection = reinterpret_cast<fp_ATrace_endSection>(
dlsym(lib, "ATrace_endSection"));
_ATrace_isEnabled = reinterpret_cast<fp_ATrace_isEnabled>(
dlsym(lib, "ATrace_isEnabled"));
} LOGI("lib=%p, _ATrace_beginSection=%p, _ATrace_endSection=%p, _ATrace_isEnabled=%p, ATrace_isEnabled()=%d",
lib, _ATrace_beginSection, _ATrace_endSection, _ATrace_isEnabled, _ATrace_isEnabled()); _bInit = true;
}

Systrace(Native)

(2)一个巧妙的利用——编译器自动插桩

利用gcc的 “-finstrument-functions” 可以实现自动化插桩。关键代码如下:

 // utils.h
extern "C" { void __cyg_profile_func_enter( void *, void * ) __attribute__ ((no_instrument_function));
void __cyg_profile_func_exit( void *, void * ) __attribute__ ((no_instrument_function)); } // utils.cpp
#define DUMP(func, call) LOGI("[KLH]%s: func = %p, called by = %p\n", __FUNCTION__, func, call) extern "C" { void __cyg_profile_func_enter(void *this_func, void *call_site) {
//DUMP(this_func, call_site);
char addr[];
snprintf(addr, , "%p", this_func); ScopedTrace::init(); if ( != _ATrace_isEnabled && _ATrace_isEnabled()) {
_ATrace_beginSection(addr);
}
} void __cyg_profile_func_exit(void *this_func, void *call_site) {
//DUMP(this_func, call_site); if ( != _ATrace_isEnabled && _ATrace_isEnabled()) {
_ATrace_endSection();
}
} }

自动插桩实现追踪

这里关键的点是需要extern "C"声明原型。

(3)编写一套Python(其他语言也行)工具来翻译加工自动插桩的函数(运行时)地址为函数名。翻译的方法为:

. 获取进程pid:
adb shell ps | adb shell grep [进程名]
分解字符串第一个即是 . 获取进程maps
adb shell cat /proc/[pid]/maps . 提取trace.html(Systrace报告)中所有运行时地址列表,准备翻译 . 提取所有so的地址段,以此来确定运行时地址所属so及内部相对地址 . 提取so内部相对地址与符号映射表
readelf -sW libXXX.so . 根据4、5翻译出运行时地址对应的函数符号,并翻译出函数名
c++filt -n [symbol] . trace.html中线程名缺失的问题可以进一步补充(如果线程中途退出则可能遗漏):
adb shell cat /proc/pid/task
[foreach] tid
adb shell cat /proc/pid/task/tid/status
得到Name:字段就是线程名称

3 原理

Android之Systrace的更多相关文章

  1. Android性能优化之Systrace工具介绍(一) _&& Systrace生成的trace.html打开空白或者打不开的解决办法

    1.必须用Chrome打开 2.在mac电脑上,可能Chrome打开也是空白,解决办法是:在chrome地址栏中输入”chrome:tracing”,然后点击load按钮load你的trace.htm ...

  2. Android性能优化工具之Systrace

    本文大部分内容来自:http://www.androidperformance.com/android-performance-tools-systrace-1.html?utm_source=tui ...

  3. Android 常用的性能分析工具详解:GPU呈现模式, TraceView, Systrace, HirearchyViewer(转)

    此篇将重点介绍几种常用的Android性能分析工具: 一.Logcat 日志 选取Tag=ActivityManager,可以粗略地知道界面Displaying的时间消耗.当我们打开一个Activit ...

  4. Android性能分析Systrace工具

    一.概述 保证系统流畅度,也就是保证系统能连续不间断地提供每秒60帧的运行状态.当出现掉帧时(也可称为Jank),需要知道当前整个系统所处的状态,systrace便是最佳的选择,它能手机检测Andro ...

  5. Android 性能优化(25)*性能工具之「Systrace」Analyzing UI Performance with Systrace:用Systrace得到ui性能报告

    Analyzing UI Performance with Systrace In this document Overview 简介 Generating a Trace  生成Systrace文件 ...

  6. android性能測试systrace

    一:简单介绍 systrace 是 Android4.1 引入的一套用于做性能分析的工具. 基于 Linux 内核的 ftrace 机制(用于跟踪 Linux 内核的函数调用), 能够输出各个线程当前 ...

  7. Android:通过systrace进行性能分析

    一.Systrace 简介 Systrace 允许您在系统级别(如SurfaceFlinger.WindowManagerService等Framework部分关键模块.服务.View系统等)收集和检 ...

  8. Android UI性能测试——使用 Systrace 查找问题

    一 官方文档翻译 官文地址:https://developer.android.com/studio/command-line/systrace systrace命令允许您在系统级别上收集和检查所有运 ...

  9. [Android systrace系列] systrace入门第一式

    转载请注明出处:https://www.cnblogs.com/zzcperf/p/13978915.html Android systrace是分析性能问题最称手的工具之一,可以提供丰富的手机运行信 ...

随机推荐

  1. zabbix的日志监控

    前提条件是该日志文件对于启动zabbix agent进程的用户开启了可读权限,而且该日志的路径对该用户开方x权限,让能取到这个日志文件 第一个参数可以不用引号,前提是zabbix用户可以进入文件路径, ...

  2. leetcode解题报告(21):Majority Element

    描述 Given an array of size n, find the majority element. The majority element is the element that app ...

  3. 块状链表 codevs 2333弹飞绵羊

    块状链表,分块处理,先预处理每一个点跳到下一个块 跳到哪,步数.然后修改的时候,修该那一个块即可 #include<cstdio>#include<cmath>int a[20 ...

  4. P3410 拍照

    漂亮小姐姐点击就送:https://www.luogu.org/problemnew/show/P3410 题目描述 小B有n个下属,现小B要带着一些下属让别人拍照. 有m个人,每个人都愿意付给小B一 ...

  5. ZR#997

    ZR#997 解法: 找找规律就出来了,全场最简单的一道题. CODE: #include<iostream> #include<cstdio> #include<cst ...

  6. radio回显的

    JS回显 var jksp=document.getElementsByName("spjdcyModel.jksp"); // alert(jksp[0].value); //a ...

  7. 实现一个简单的Tomcat

    实现一个简单的Tomcat 1. Tomcat作用 我们的web应用会运行在Tomcat中,那么显然请求必定是先到达Tomcat的,Tomcat对于请求实际上会进行如下的处理: 提供Socket服务: ...

  8. 2018-2019-2 20165312《网络攻防技术》Exp7 网络欺诈防范

    2018-2019-2 20165312<网络攻防技术>Exp7 网络欺诈防范 目录 一.相关知识点总结 二.实验内容 三.实验步骤 四.实验总结及问题回答 五.实验中遇到的问题及解决方法 ...

  9. Fiddler is not capturing web request from Firefox

    Fiddler is not capturing web request from Firefox You can also get the FiddlerHook plug in for Firef ...

  10. linux内核中的__cpu_suspend是在哪里实现的呀?

    1. 内核版本 4.19 2. 在arch/arm/kernel/sleep.S中实现如下: /* * Save CPU state for a suspend. This saves the CPU ...