跳转至

Matrix-TraceCanary解析

正如Matrix系列开篇——微信APM Matrix解析所言,TraceCanary分为帧率监控、慢方法监控、ANR监控以及启动耗时这4个功能。这些功能基本都需要插桩去trace过程中每个函数的调用。
在插桩之外,我们还需要监控主线程MessageQueue(LooperMonitor)以及Choreographer(UIThreadMonitor),从里面抽象出一个可以通知Message开始执行、执行完成、Choreographer开始渲染的接口(LooperObserver)。基于这个接口,我们可以开始各项监控的实现。

确实,本章的重点就是解读 插桩插件 以及 LooperMonitor、UIThreadMonitor 是如何实现的。在理解上面这些组件的原理之后,基于此的监控的实现也就非常好理解了。

当然,在正式开始分析源码之前,我们先看一下Sample是如何初始化Matrix以及如何启动TracePlugin的。

1. 从Sample中看Plugin的启动

Matrix的初始化代码位于Sample的Application中,这里根据配置创建了TracePlugin、ResourcePlugin、IOCanaryPlugin以及SQLiteLintPlugin后添加到了Matrix.Builder中:

matrix/samples/sample-android/app/src/main/java/sample/tencent/matrix/MatrixApplication.java

public class MatrixApplication extends Application {
    ...
    @Override
    public void onCreate() {
        super.onCreate();
        ...

        Matrix.Builder builder = new Matrix.Builder(this);
        builder.patchListener(new TestPluginListener(this));

        //trace
        TraceConfig traceConfig = new TraceConfig.Builder()
                .dynamicConfig(dynamicConfig)
                .enableFPS(fpsEnable)
                .enableEvilMethodTrace(traceEnable)
                .enableAnrTrace(traceEnable)
                .enableStartup(traceEnable)
                .splashActivities("sample.tencent.matrix.SplashActivity;")
                .isDebug(true)
                .isDevEnv(false)
                .build();

        TracePlugin tracePlugin = (new TracePlugin(traceConfig));
        builder.plugin(tracePlugin);

        if (matrixEnable) {

            //resource
            ...
            builder.plugin(new ResourcePlugin(resourceConfig));
            ResourcePlugin.activityLeakFixer(this);

            //io
            IOCanaryPlugin ioCanaryPlugin = new IOCanaryPlugin(new IOConfig.Builder()
                    .dynamicConfig(dynamicConfig)
                    .build());
            builder.plugin(ioCanaryPlugin);

            // prevent api 19 UnsatisfiedLinkError
            //sqlite
            ...
            builder.plugin(new SQLiteLintPlugin(sqlLiteConfig));
        }

        Matrix.init(builder.build());

        //start only startup tracer, close other tracer.
        tracePlugin.start();
    }
}

在第45行中,builder.build()会创建Matrix实例并调用所有添加过的Plugin的init方法,而Matrix.init方法仅仅是保存了这个Matrix实例而已。最后在48行中,启动了TracePlugin插件。

Matrix实例创建的相关代码如下:

matrix/matrix/matrix-android/matrix-android-lib/src/main/java/com/tencent/matrix/Matrix.java

public class Matrix {
    ...

    private Matrix(Application app, PluginListener listener, HashSet<Plugin> plugins) {
        this.application = app;
        this.pluginListener = listener;
        this.plugins = plugins;
        AppActiveMatrixDelegate.INSTANCE.init(application);
        for (Plugin plugin : plugins) {
            plugin.init(application, pluginListener);
            pluginListener.onInit(plugin);
        }

    }

    ...

    public static Matrix init(Matrix matrix) {
        if (matrix == null) {
            throw new RuntimeException("Matrix init, Matrix should not be null.");
        }
        synchronized (Matrix.class) {
            if (sInstance == null) {
                sInstance = matrix;
            } else {
                MatrixLog.e(TAG, "Matrix instance is already set. this invoking will be ignored");
            }
        }
        return sInstance;
    }

    ...

    public static class Builder {
        ...
        private HashSet<Plugin> plugins = new HashSet<>();
        ...

        public Matrix build() {
            if (pluginListener == null) {
                pluginListener = new DefaultPluginListener(application);
            }
            return new Matrix(application, pluginListener, plugins);
        }

    }
}

2. TracePlugin

在上面的Matrix的启动流程中,上面我们看到了TracePlugin的init方法以及start方法的调用。下面我们回归本章的主题,看看TracePlugin到底干了些什么。

首先看看TracePlugin初始化相关的代码:

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/TracePlugin.java

public class TracePlugin extends Plugin {
    private static final String TAG = "Matrix.TracePlugin";

    private final TraceConfig traceConfig;
    private EvilMethodTracer evilMethodTracer;
    private StartupTracer startupTracer;
    private FrameTracer frameTracer;
    private AnrTracer anrTracer;

    public TracePlugin(TraceConfig config) {
        this.traceConfig = config;
    }

    @Override
    public void init(Application app, PluginListener listener) {
        super.init(app, listener);
        MatrixLog.i(TAG, "trace plugin init, trace config: %s", traceConfig.toString());
        if (Build.VERSION.SDK_INT < Build.VERSION_CODES.JELLY_BEAN) {
            MatrixLog.e(TAG, "[FrameBeat] API is low Build.VERSION_CODES.JELLY_BEAN(16), TracePlugin is not supported");
            unSupportPlugin();
            return;
        }

        anrTracer = new AnrTracer(traceConfig);

        frameTracer = new FrameTracer(traceConfig);

        evilMethodTracer = new EvilMethodTracer(traceConfig);

        startupTracer = new StartupTracer(traceConfig);
    }

    @Override
    public void start() {
        super.start();
        if (!isSupported()) {
            MatrixLog.w(TAG, "[start] Plugin is unSupported!");
            return;
        }
        MatrixLog.w(TAG, "start!");
        Runnable runnable = new Runnable() {
            @Override
            public void run() {

                if (!UIThreadMonitor.getMonitor().isInit()) {
                    try {
                        UIThreadMonitor.getMonitor().init(traceConfig);
                    } catch (java.lang.RuntimeException e) {
                        MatrixLog.e(TAG, "[start] RuntimeException:%s", e);
                        return;
                    }
                }

                AppMethodBeat.getInstance().onStart();

                UIThreadMonitor.getMonitor().onStart();

                anrTracer.onStartTrace();

                frameTracer.onStartTrace();

                evilMethodTracer.onStartTrace();

                startupTracer.onStartTrace();
            }
        };

        if (Thread.currentThread() == Looper.getMainLooper().getThread()) {
            runnable.run();
        } else {
            MatrixLog.w(TAG, "start TracePlugin in Thread[%s] but not in mainThread!", Thread.currentThread().getId());
            MatrixHandlerThread.getDefaultMainHandler().post(runnable);
        }
    }
}

init方法中,对低于JB的设备不启用TracePlugin,并创建了ANR监控、帧率监控、慢方法监控以及启动耗时这4个功能的Tracer。在start中,会在主线程中启动UIThreadMonitorAppMethodBeat以及四大Tracer。

TracePlugin的功能完全体现在了四大Tracer中,UIThreadMonitorAppMethodBeat是这些Tracer起作用的基石。理解了这些基石,四大Tracer的分析就非常简单了。所以下面我们先分析分析UIThreadMonitorAppMethodBeat

此外,基于这两大基石,我们还可以写出更多好玩的东西:比如 后台渲染的检测UIThreadMonitor判断后台渲染是否产生,由AppMethodBeat可以获取引发后台渲染的堆栈信息。

2.1 UIThreadMonitor

一句话简述UIThreadMonitor

通过设置Looper中的printer,来判断Message的执行起止时间。然后hook Choreographer中的input animation traversal回调数组,向其中添加Runnable来获取每个操作的耗时。最后将这些数据抛出给各个Tracer作为判断的依据。

获取线程中每个Message的执行起止时间是在Matrix中是LooperMonitor类来实现的,UIThreadMonitor向该类注册一个回调,由此在对应回调中进行对应的操作。

LooperMonitor的实现原理我们下一节在谈,目前我们知道它干了啥就行。我们接着看UIThreadMonitor#init方法:

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/core/UIThreadMonitor.java

public void init(TraceConfig config) {
    if (Thread.currentThread() != Looper.getMainLooper().getThread()) {
        throw new AssertionError("must be init in main thread!");
    }
    this.config = config;
    choreographer = Choreographer.getInstance();
    callbackQueueLock = reflectObject(choreographer, "mLock");
    callbackQueues = reflectObject(choreographer, "mCallbackQueues");

    addInputQueue = reflectChoreographerMethod(callbackQueues[CALLBACK_INPUT], ADD_CALLBACK, long.class, Object.class, Object.class);
    addAnimationQueue = reflectChoreographerMethod(callbackQueues[CALLBACK_ANIMATION], ADD_CALLBACK, long.class, Object.class, Object.class);
    addTraversalQueue = reflectChoreographerMethod(callbackQueues[CALLBACK_TRAVERSAL], ADD_CALLBACK, long.class, Object.class, Object.class);
    frameIntervalNanos = reflectObject(choreographer, "mFrameIntervalNanos");

    LooperMonitor.register(new LooperMonitor.LooperDispatchListener() {
        @Override
        public boolean isValid() {
            return isAlive;
        }

        @Override
        public void dispatchStart() {
            super.dispatchStart();
            UIThreadMonitor.this.dispatchBegin();
        }

        @Override
        public void dispatchEnd() {
            super.dispatchEnd();
            UIThreadMonitor.this.dispatchEnd();
        }

    });
    this.isInit = true;
    MatrixLog.i(TAG, "[UIThreadMonitor] %s %s %s %s %s frameIntervalNanos:%s", callbackQueueLock == null, callbackQueues == null, addInputQueue == null, addTraversalQueue == null, addAnimationQueue == null, frameIntervalNanos);

    if (config.isDevEnv()) {
        addObserver(new LooperObserver() {
            @Override
            public void doFrame(String focusedActivityName, long start, long end, long frameCostMs, long inputCost, long animationCost, long traversalCost) {
                MatrixLog.i(TAG, "activityName[%s] frame cost:%sms [%s|%s|%s]ns", focusedActivityName, frameCostMs, inputCost, animationCost, traversalCost);
            }
        });
    }
}

UIThreadMonitor#init方法主要干了两件事:

  1. 反射获取ChoreographerCALLBACK_INPUTCALLBACK_ANIMATIONCALLBACK_TRAVERSAL三种类型的CallbackQueueaddCallbackLocked方法的句柄。
  2. LooperMonitor注册Message执行开始的回调、执行结束的回调。这里的Message是指主线程中发生的所有Message,包括App自己的以及Framework中的,Choreographer中的自然也可以捕获到。

下面看看UIThreadMonitor#start方法,在里面初始化了需要检测的三种CallbackQueue的各种记录数组:

  • callbackExist:记录是否已经向该类型的CallbackQueue添加了Runnable,避免重复添加
  • queueStatus:记录CallbackQueue中添加的Runnable的运行状态,分为默认状态(DO_QUEUE_DEFAULT)、已经添加的状态(DO_QUEUE_BEGIN)、运行结束的状态(DO_QUEUE_END)
  • queueCost:记录上面某种类型的Runnable的执行起始的耗时,这可以反映出当前这一次执行CallbackQueue里面的任务耗时有多久。

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/core/UIThreadMonitor.java

@Override
public synchronized void onStart() {
    if (!isInit) {
        throw new RuntimeException("never init!");
    }
    if (!isAlive) {
        this.isAlive = true;
        synchronized (this) {
            MatrixLog.i(TAG, "[onStart] callbackExist:%s %s", Arrays.toString(callbackExist), Utils.getStack());
            callbackExist = new boolean[CALLBACK_LAST + 1];
        }
        queueStatus = new int[CALLBACK_LAST + 1];
        queueCost = new long[CALLBACK_LAST + 1];
        addFrameCallback(CALLBACK_INPUT, this, true);
    }
}

UIThreadMonitor#onStart方法中,最后调用addFrameCallback方法将一个Runnable(自己)插到了INPUT类型的CallbackQueue的头部。CallbackQueue是一个单链表组织起来的队列,里面按照时间从小到大进行组织,详细代码可以查看Choreographer#CallbackQueue

下面我们接着看一下addFrameCallback的实现,这里首先会判断某种type类型的callback是否已经添加,UIThreadMonitor是否已经启动等等检查,然后根据type取得需要invoke的方法句柄,然后调用该方法并设置callbackExist标志位。

在第26行中我们注意到对入参isAddHeader做出的值转换——!isAddHeader ? SystemClock.uptimeMillis() : -1:也就是说如果isAddHeader为true,这里的值就是-1,会在CallbackQueue执行时首先执行(系统内部不会将这里的time设置为一个负值,所以无论何时,这里都将会是第一个执行的);否则的话,传的是当前的时间戳,会根据值插入到单链表队列中。

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/core/UIThreadMonitor.java

private synchronized void addFrameCallback(int type, Runnable callback, boolean isAddHeader) {
    if (callbackExist[type]) {
        MatrixLog.w(TAG, "[addFrameCallback] this type %s callback has exist! isAddHeader:%s", type, isAddHeader);
        return;
    }

    if (!isAlive && type == CALLBACK_INPUT) {
        MatrixLog.w(TAG, "[addFrameCallback] UIThreadMonitor is not alive!");
        return;
    }
    try {
        synchronized (callbackQueueLock) {
            Method method = null;
            switch (type) {
                case CALLBACK_INPUT:
                    method = addInputQueue;
                    break;
                case CALLBACK_ANIMATION:
                    method = addAnimationQueue;
                    break;
                case CALLBACK_TRAVERSAL:
                    method = addTraversalQueue;
                    break;
            }
            if (null != method) {
                method.invoke(callbackQueues[type], !isAddHeader ? SystemClock.uptimeMillis() : -1, callback, null);
                callbackExist[type] = true;
            }
        }
    } catch (Exception e) {
        MatrixLog.e(TAG, e.toString());
    }
}

上面就是onStart方法干的事情,归根结底就是向Choreograpger注册了一个回调(即UIThreadMonitor自身),这样下次Vsync信号来到时,就会触发这个callback(UIThreadMonitor#run方法)。

然后我们看看UIThreadMonitor#run里面的代码,这里面涉及到三个新的方法:

  1. doFrameBegin:设置isBelongFrame标志位为true,这标志着当前Frame已经被纳入了统计
  2. doQueueBegin:更新对应type的queueStatus标志位为DO_QUEUE_BEGIN,并用queueCost[type]记下此时的时间
  3. doQueueEnd:更新对应type的queueStatus标志位为DO_QUEUE_END,并用当前时间减去queueCost[type]的时间,这个时间为当前frame执行时此type的CallbackQueue执行的总耗时,记为queueCost[type]

因此,UIThreadMonitor#run就很好解释了:先声明该frame正在被统计,然后计算出CALLBACK_INPUT的队列执行耗时、CALLBACK_ANIMATION的队列执行耗时,以及CALLBACK_TRAVERSAL执行开始了。

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/core/UIThreadMonitor.java

@Override
public void run() {
    final long start = System.nanoTime();
    try {
        doFrameBegin(token);
        doQueueBegin(CALLBACK_INPUT);

        addFrameCallback(CALLBACK_ANIMATION, new Runnable() {

            @Override
            public void run() {
                doQueueEnd(CALLBACK_INPUT);
                doQueueBegin(CALLBACK_ANIMATION);
            }
        }, true);

        addFrameCallback(CALLBACK_TRAVERSAL, new Runnable() {

            @Override
            public void run() {
                doQueueEnd(CALLBACK_ANIMATION);
                doQueueBegin(CALLBACK_TRAVERSAL);
            }
        }, true);

    } finally {
        if (config.isDevEnv()) {
            MatrixLog.d(TAG, "[UIThreadMonitor#run] inner cost:%sns", System.nanoTime() - start);
        }
    }
}

那,为什么这里没有调用doQueueEnd(CALLBACK_TRAVERSAL)。我们研究Choreographer发现,在CALLBACK_TRAVERSAL之后还有一个CALLBACK_COMMIT,我们向CALLBACK_COMMIT这个队列添加一个callback就可以在合理的位置调用doQueueEnd(CALLBACK_TRAVERSAL)了。 但是很不幸,CALLBACK_COMMIT在Android 6.0及以后才会有。

为了兼容更早的版本,我们得想出其他办法:还记得上面提到的LooperMonitor吗,我们提到过LooperMonitor可以捕获到Message的执行的起始。Choreographer中的Vsync信号触发各种callback也是通过Android的消息机制来实现的,且该Message在执行完各种CallbackQueue就结束了,某种程度上来说,以Message的执行结束时间作CALLBACK_TRAVERSAL的结束时间也是可以的。

所以我们接着就来到了UIThreadMonitor#dispatchEnd方法:

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/core/UIThreadMonitor.java

private void dispatchEnd() {

    if (isBelongFrame) {
        doFrameEnd(token);
    }

    long start = token;
    long end = SystemClock.uptimeMillis();

    synchronized (observers) {
        for (LooperObserver observer : observers) {
            if (observer.isDispatchBegin()) {
                observer.doFrame(AppMethodBeat.getVisibleScene(), token, SystemClock.uptimeMillis(), isBelongFrame ? end - start : 0, queueCost[CALLBACK_INPUT], queueCost[CALLBACK_ANIMATION], queueCost[CALLBACK_TRAVERSAL]);
            }
        }
    }

    dispatchTimeMs[3] = SystemClock.currentThreadTimeMillis();
    dispatchTimeMs[1] = SystemClock.uptimeMillis();

    AppMethodBeat.o(AppMethodBeat.METHOD_ID_DISPATCH);

    synchronized (observers) {
        for (LooperObserver observer : observers) {
            if (observer.isDispatchBegin()) {
                observer.dispatchEnd(dispatchTimeMs[0], dispatchTimeMs[2], dispatchTimeMs[1], dispatchTimeMs[3], token, isBelongFrame);
            }
        }
    }

}

在方法的最开始,如果isBelongFrame标志位为true,表明是UIThreadMonitor#run方法已经执行过了,因此需要闭合整个frame的监控,闭合的这部分代码在doFrameEnd中。
然后会调用LooperObserver.doFrame方法通知观察者当前帧的耗时情况,调用LooperObserver.dispatchEnd通知观察者Message执行已经结束了。

这里没啥好说的,我们看一下闭合frame监控的doFrameEnd方法,这里调用了doQueueEnd(CALLBACK_TRAVERSAL)真正闭合了frame监控,然后恢复了queueStatus数组的状态,最后又调用了addFrameCallback(CALLBACK_INPUT, this, true)方法开始监控下一帧。嗯,复读机又回来了~:

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/core/UIThreadMonitor.java

private void doFrameEnd(long token) {

    doQueueEnd(CALLBACK_TRAVERSAL);

    for (int i : queueStatus) {
        if (i != DO_QUEUE_END) {
            queueCost[i] = DO_QUEUE_END_ERROR;
            if (config.isDevEnv) {
                throw new RuntimeException(String.format("UIThreadMonitor happens type[%s] != DO_QUEUE_END", i));
            }
        }
    }
    queueStatus = new int[CALLBACK_LAST + 1];

    addFrameCallback(CALLBACK_INPUT, this, true);

    this.isBelongFrame = false;
}

bug

值得注意的是,这里有个bug, 当isBelongFrame为true时,会闭合frame的监控,也就是doFrameEnd干的事儿。但是在该方法的最后,将isBelongFrame复位为false了。所以通知LooperObserver#doFrame以及LooperObserver#dispatchEnd中涉及到的值,实际上都是false。
在实际应用时,我们应该注意下,去修复这个问题。博主目前注意到,matrix在hotfix/0.6.x分支上已经修复了这个问题。

说完了UIThreadMonitor#dispatchEnd方法,我们也顺便说说与之匹配的孪生方法UIThreadMonitor#dispatchBegin

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/core/UIThreadMonitor.java

private void dispatchBegin() {
    token = dispatchTimeMs[0] = SystemClock.uptimeMillis();
    dispatchTimeMs[2] = SystemClock.currentThreadTimeMillis();
    AppMethodBeat.i(AppMethodBeat.METHOD_ID_DISPATCH);

    synchronized (observers) {
        for (LooperObserver observer : observers) {
            if (!observer.isDispatchBegin()) {
                observer.dispatchBegin(dispatchTimeMs[0], dispatchTimeMs[2], token);
            }
        }
    }
}

实际上该方法也很简单,保存下Message执行开始时的时间戳,然后调用LooperObserver#dispatchBegin方法回调出去。
dispatchTimeMs是一个长度为4的long数组,0、1分别表示的是Message执行开始、结束的uptimeMillis;2、3分别表示的是Message执行开始、结束的currentThreadTimeMillis

至此,UIThreadMonitor的解析已经完毕,我们可以小结一下其作用:

  1. 在主线程中Message执行开始时,调用LooperObserver#dispatchBegin(long beginMs, long cpuBeginMs, long token)通知外部
  2. 在主线程中每个Message执行结束时,调用LooperObserver#doFrame(String focusedActivityName, long start, long end, long frameCostMs, long inputCostNs, long animationCostNs, long traversalCostNs)通知外部。在参数中,可以以frameCostMs的值反推isBelongFrame值。
  3. 在主线程中Message执行完毕时,调用LooperObserver#dispatchEnd(long beginMs, long cpuBeginMs, long endMs, long cpuEndMs, long token, boolean isBelongFrame)通知外部

2.2 LooperMonitor

LooperMonitor的主要作用就是在Message执行前后回调出对应的方法。其实现原理是通过设置Looper#mLogging这个字段,让Message在执行前后打印出日志,然后根据日志的特点来判断是开始执行还是执行结束。可以看下Looper#loop方法的代码片段:

android/os/Looper.java

public static void loop() {
    ...
    for (;;) {
        Message msg = queue.next(); // might block
        ...
        // This must be in a local variable, in case a UI event sets the logger
        final Printer logging = me.mLogging;
        if (logging != null) {
            logging.println(">>>>> Dispatching to " + msg.target + " " +
                    msg.callback + ": " + msg.what);
        }
        ...
        try {
            msg.target.dispatchMessage(msg);
            ...
        } finally {
            ...
        }
        ...
        if (logging != null) {
            logging.println("<<<<< Finished to " + msg.target + " " + msg.callback);
        }
        ...
    }
}

下面看看LooperMonitor的初始化相关的代码。
首先LooperMonitor是一个饥汉单例的实现,其作用对象是主线程的Looper。
在LooperMonitor创建之后,首先调用resetPrinter方法保存原始的printer,然后使用LooperPrinter来装饰原始的printer并设置到Looper中,这样我们判断print的日志就知道Message的执行起始。
最后,向MessageQueue中添加了一个IdleHandler,在对应的queueIdle方法中会周期性(60s)的调用resetPrinter方法来保证Looper中的printer对象是我们自定义的LooperPrinter

/matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/core/LooperMonitor.java

public class LooperMonitor implements MessageQueue.IdleHandler {
    ...
    private static final LooperMonitor mainMonitor = new LooperMonitor();

    public LooperMonitor(Looper looper) {
        Objects.requireNonNull(looper);
        this.looper = looper;
        resetPrinter();
        addIdleHandler(looper);
    }

    private LooperMonitor() {
        this(Looper.getMainLooper());
    }

    private static boolean isReflectLoggingError = false;

    private synchronized void resetPrinter() {
        Printer originPrinter = null;
        try {
            if (!isReflectLoggingError) {
                originPrinter = ReflectUtils.get(looper.getClass(), "mLogging", looper);
                if (originPrinter == printer && null != printer) {
                    return;
                }
            }
        } catch (Exception e) {
            isReflectLoggingError = true;
            Log.e(TAG, "[resetPrinter] %s", e);
        }

        if (null != printer) {
            MatrixLog.w(TAG, "maybe thread:%s printer[%s] was replace other[%s]!",
                    looper.getThread().getName(), printer, originPrinter);
        }
        looper.setMessageLogging(printer = new LooperPrinter(originPrinter));
        if (null != originPrinter) {
            MatrixLog.i(TAG, "reset printer, originPrinter[%s] in %s", originPrinter, looper.getThread().getName());
        }
    }

    private synchronized void addIdleHandler(Looper looper) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            looper.getQueue().addIdleHandler(this);
        } else {
            try {
                MessageQueue queue = ReflectUtils.get(looper.getClass(), "mQueue", looper);
                queue.addIdleHandler(this);
            } catch (Exception e) {
                Log.e(TAG, "[removeIdleHandler] %s", e);
            }
        }
    }
    ...
}

下面我们来到了LooperPrinter对象,直接看其println方法的实现,我们可以发现直接判断起始字符为不为>就可以知道是onDispatchStart还是onDispatchEnd

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/core/LooperMonitor.java

class LooperPrinter implements Printer {
    public Printer origin;
    boolean isHasChecked = false;
    boolean isValid = false;

    LooperPrinter(Printer printer) {
        this.origin = printer;
    }

    @Override
    public void println(String x) {
        if (null != origin) {
            origin.println(x);
            if (origin == this) {
                throw new RuntimeException(TAG + " origin == this");
            }
        }

        if (!isHasChecked) {
            isValid = x.charAt(0) == '>' || x.charAt(0) == '<';
            isHasChecked = true;
            if (!isValid) {
                MatrixLog.e(TAG, "[println] Printer is inValid! x:%s", x);
            }
        }

        if (isValid) {
            dispatch(x.charAt(0) == '>', x);
        }

    }
}

private void dispatch(boolean isBegin, String log) {

    for (LooperDispatchListener listener : listeners) {
        if (listener.isValid()) {
            if (isBegin) {
                if (!listener.isHasDispatchStart) {
                    listener.onDispatchStart(log);
                }
            } else {
                if (listener.isHasDispatchStart) {
                    listener.onDispatchEnd(log);
                }
            }
        } else if (!isBegin && listener.isHasDispatchStart) {
            listener.dispatchEnd();
        }
    }
}

3. AppMethodBeat

关于Plugin

Matrix master分支的插件(0.6.5版本)代码不支持R8以及AGP4.x。在feature分支有一个upgrade-agp-4.1-wip分支对此做了支持。但是不影响我们对其原理的分析解读。
但是这部分源码颇费笔墨,所以在Matrix-ASM插桩插件解析中进行解读。

AppMethodBeat会被Plugin在编译时进行调用,调用位置为Java方法的出入口以及Activity#onWindowFocusChange,Plugin会在这些位置分别调用其i(int methodId)o(int methodId)以及at方法。
对于i/o方法里面的参数methodId,Plugin在插桩时会为每一个函数生成一个唯一的id,保证通过这个id可以找到对应的方法,methodId可以在编译文件app/build/outputs/mapping/debug/methodMapping.txt中得到。

下面是methodMapping文件中的示例: 1,1,sample.tencent.matrix.trace.TestTraceFragmentActivity ()V 2,1,sample.tencent.matrix.trace.TestFpsActivity ()V
上面分别表示:methodId,accessFlag,className methodName [desc]

插桩过程中会过滤一些不需要插桩的函数,结果输出在app/build/outputs/mapping/debug/methodMapping.txt中,其格式为className methodName [desc]

插桩的这部分源码在matrix/matrix-android/matrix-gradle-plugin/src/main/java/com/tencent/matrix/trace/MethodTracer.java中,后面在解读Matrix Plugin时会进行详细说明。BTW,其实插桩插件流程总体可以分为两步:第一步收集工程中的class以及jar包中的class,在插桩完毕后进行回写,这一步都比较通用;第二步就是调用ASM进行插桩,这一部分才与需求相关。

AppMethodBeat#iAppMethodBeat#o会将函数调用的i/o标志、methodId以及时间存到sBuffer = new long[100 * 10000]数组中,这个数组消耗内存约为8bytes * 100 * 10000 = 800_0000bytes = 7812.5kb = 7.629394531mb。这部分 内存消耗还是有点大的,是一个副作用吧

我们说到,后面各种Tracer都是分析这个sBuffer数组来得到的函数调用堆栈,那么这个数组里面保存的数据有怎么样的格式呢?

Matrix在编译期会对全局的函数进行插桩,在运行期间每个函数的执行前后都会调用 AppMethodBeat.i/o 的方法,如果是在主线程中执行,则在函数的执行前后获取当前距离 MethodBeat 模块初始化的时间 offset(为了压缩数据,存进一个long类型变量中),并将当前执行的是 AppMethodBeat i或者o、mehtod id 及时间 offset,存放到一个 long 类型变量中,记录到一个预先初始化好的数组 long[] 中 index 的位置(预先分配记录数据的 buffer 长度为 100w,内存占用约 7.6M)。数据存储如下图1

matrix_app_method_beat_sbuffer

AppMethodBeat.i/o主要干的就是上面的这个事儿;在AppMethodBeat.at方法中,会在Activity#onWindowFocusChange时调用IAppMethodBeatListener#onActivityFocused方法。

小结一下,Matrix会在编译时对函数进行插桩,这样在运行期间每个函数的执行前后都会调用 AppMethodBeat.i/o 的方法,这些方法的调用记录会被数组保存起来,供后面各种Tracer进行函数调用堆栈分析。并且会在Activity#onWindowFocusChange处插入 AppMethodBeat.at 方法,当Activity获得焦点时调用回调通知外部。

4. 各种Tracer

4.1 帧率监控FrameTracer

FrameTracer的实现依赖于UIThreadMonitor中抛出来的LooperObserver#doFrame回调。该回调的方法声明如下:

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/listeners/LooperObserver.java

public void doFrame(String focusedActivityName, long start, long end, long frameCostMs, long inputCostNs, long animationCostNs, long traversalCostNs)

各个参数解释如下:

  • focusedActivityName
    正在显示的Activity名称
  • start
    Message执行前的时间
  • end
    Message执行完毕,调用LooperObserver#doFrame时的时间
  • frameCostMs
    如果调用执行dispatchEnd时,UIThreadMonitor#run执行过了,那么该值为上面的end-start的值;否则为0
  • inputCostNs、animationCostNs、traversalCostNs
    执行三种CallbackQueue的耗时

下面我们看看FrameTracer是如何通过这些值计算出帧率信息的,直接看override的doFrame方法:

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/tracer/FrameTracer.java

@Override
public void doFrame(String focusedActivityName, long start, long end, long frameCostMs, long inputCostNs, long animationCostNs, long traversalCostNs) {
    if (isForeground()) {
        notifyListener(focusedActivityName, end - start, frameCostMs, frameCostMs >= 0);
    }
}

private void notifyListener(final String visibleScene, final long taskCostMs, final long frameCostMs, final boolean isContainsFrame) {
    long start = System.currentTimeMillis();
    try {
        synchronized (listeners) {
            for (final IDoFrameListener listener : listeners) {
                if (config.isDevEnv()) {
                    listener.time = SystemClock.uptimeMillis();
                }
                final int dropFrame = (int) (taskCostMs / frameIntervalMs);

                listener.doFrameSync(visibleScene, taskCostMs, frameCostMs, dropFrame, isContainsFrame);
                if (null != listener.getExecutor()) {
                    listener.getExecutor().execute(new Runnable() {
                        @Override
                        public void run() {
                            listener.doFrameAsync(visibleScene, taskCostMs, frameCostMs, dropFrame, isContainsFrame);
                        }
                    });
                }
                if (config.isDevEnv()) {
                    listener.time = SystemClock.uptimeMillis() - listener.time;
                    MatrixLog.d(TAG, "[notifyListener] cost:%sms listener:%s", listener.time, listener);
                }
            }
        }
    } finally {
        long cost = System.currentTimeMillis() - start;
        if (config.isDebug() && cost > frameIntervalMs) {
            MatrixLog.w(TAG, "[notifyListener] warm! maybe do heavy work in doFrameSync! size:%s cost:%sms", listeners.size(), cost);
        }
    }
}

FrameTracer#doFrame中会在App处于 前台 的状态下调用notifyListener方法,注意这里的taskCostMs参数值为end-start,实际上这个参数的值应该与frameCostMs的值一致。
notifyListener方法中,首先通过taskCostMs / frameIntervalMs来计算此次执行丢的帧数,记为dropFrame。也就是说单次Message的执行耗时不超过16ms,则不会产生丢帧。最后调用IDoFrameListener#doFrameSync以及IDoFrameListener#doFrameAsync执行lisetner的同步方法以及异步方法。

在有了上面计算出来的dropFrame之后,我们在后面就不太需要taskCostMs以及frameCostMs这两个值了。我们把注意力放到其他几个参数中去,有了这几个参数(visibleScene、dropFrame、isContainsFrame),后面就可以进行帧率的上报以及实时帧率的显示。

先看看帧率上报的部分代码。在FrameTracer构造的时候,Matrix向其中添加了一个FPSCollector这个实现了IDoFrameListener接口的类,用来进行帧率的收集以及上报。

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/tracer/FrameTracer.java

private class FPSCollector extends IDoFrameListener {

    private Handler frameHandler = new Handler(MatrixHandlerThread.getDefaultHandlerThread().getLooper());

    Executor executor = new Executor() {
        @Override
        public void execute(Runnable command) {
            frameHandler.post(command);
        }
    };

    private HashMap<String, FrameCollectItem> map = new HashMap<>();

    @Override
    public Executor getExecutor() {
        return executor;
    }

    @Override
    public void doFrameAsync(String visibleScene, long taskCost, long frameCostMs, int droppedFrames, boolean isContainsFrame) {
        super.doFrameAsync(visibleScene, taskCost, frameCostMs, droppedFrames, isContainsFrame);
        if (Utils.isEmpty(visibleScene)) {
            return;
        }

        FrameCollectItem item = map.get(visibleScene);
        if (null == item) {
            item = new FrameCollectItem(visibleScene);
            map.put(visibleScene, item);
        }

        item.collect(droppedFrames, isContainsFrame);

        if (item.sumFrameCost >= timeSliceMs) { // report
            map.remove(visibleScene);
            item.report();
        }
    }
}

FPSCollector以页面为key,记录每个页面的帧情况。首先调用FrameCollectItem.collect收集帧率信息。然后当页面累计事件sumFrameCost超过一定阈值(默认10s)时,进行上报。

然后我们看看重点的FrameCollectItem#collect方法:

void collect(int droppedFrames, boolean isContainsFrame) {
    long frameIntervalCost = UIThreadMonitor.getMonitor().getFrameIntervalNanos();
    sumFrameCost += (droppedFrames + 1) * frameIntervalCost / Constants.TIME_MILLIS_TO_NANO;
    sumDroppedFrames += droppedFrames;
    sumFrame++;
    if (!isContainsFrame) {
        sumTaskFrame++;
    }

    if (droppedFrames >= frozenThreshold) {
        dropLevel[DropStatus.DROPPED_FROZEN.index]++;
        dropSum[DropStatus.DROPPED_FROZEN.index] += droppedFrames;
    } else if (droppedFrames >= highThreshold) {
        dropLevel[DropStatus.DROPPED_HIGH.index]++;
        dropSum[DropStatus.DROPPED_HIGH.index] += droppedFrames;
    } else if (droppedFrames >= middleThreshold) {
        dropLevel[DropStatus.DROPPED_MIDDLE.index]++;
        dropSum[DropStatus.DROPPED_MIDDLE.index] += droppedFrames;
    } else if (droppedFrames >= normalThreshold) {
        dropLevel[DropStatus.DROPPED_NORMAL.index]++;
        dropSum[DropStatus.DROPPED_NORMAL.index] += droppedFrames;
    } else {
        dropLevel[DropStatus.DROPPED_BEST.index]++;
        dropSum[DropStatus.DROPPED_BEST.index] += (droppedFrames < 0 ? 0 : droppedFrames);
    }
}

sumFrameCost记录的是累计的帧耗时。sumDroppedFrames记录的是累计丢帧数。sumFrame是累计帧数。在记录这些数之后,按照丢帧的数值,记录到对应的丢帧状态数组中。在记录完毕之后,会在适当的时间被调用report方法进行上报,上报的代码就不是重点了。

上面就是TrameTracer中帧率上报的代码分析,除开具体的计算部分之外,其他代码非常简明扼要。但就是计算部分的代码,我始终觉得有很大的bug。

isContainsFrame始终为true

我们回到UIThreadMonitor#dispatchEnd中调用LooperObserver#doFrame的位置,我们假设前面提到的isBelongFrame始终为false的bug已经解决了,再看isBelongFrame ? end - start : 0这段代码,这段代码的值肯定始终是>= 0的。然后我们看FrameTracer#doFrame中的对于isContainsFrame参数的计算:frameCostMs >= 0,毫无疑问,该值也是恒真的。
这就导致在FrameCollectItem#collectsumTaskFrame始终不能自增,导致在report时dropTaskFrameSum始终为0。
matrix目前在hotfix/0.6.x分支上修复了一部分bug,上面这个bug也修复过了。此外,该分支上额外还发现了battery、memory、thread等部分的文件夹。按照官方的说法,这些内容还不是很稳定,不过后续有需要还是可以参考下。

sumFrameCost计算真的合理吗

我们说到sumFrameCost是统计的累计帧耗时,那么当isContainsFrame为false时,也就是执行其他Message时,sumFrameCost也会累积?
比如我们在demo中直接在主Handler循环delay 1ms执行自己,那么最终也会由UIThreadMonitor捕获到该Message并最后传到FrameCollectItem#collect中。尽管没有触发任何UI操作,但还是按照frame的message进行统计了。
我们可以执行下面代码,发现很快就触发了上报。

final Handler handler = new Handler();
handler.postDelayed(new Runnable() {
    @Override
    public void run() {
        handler.postDelayed(this, 1);
    }
}, 1);
感觉这里先看看是不是isContainsFrame,然后在累加比较合适。

开箱即用

matrix-trace-canary中有一个FrameDecorator的类,可以悬浮窗展示实时帧率,开箱即用,无需自己写逻辑。其底层实现与FrameTracer类似。

4.2 慢方法监控EvilMethodTracer

与FrameTracer类似,EvilMethodTracer的实现也依赖于UIThreadMonitor中抛出来的LooperObserver接口里面的dispatchBegindoFramedispatchEnd三个方法。

dispatchBegin方法中,记录下AppMethodBeat中目前的index,记为start;
dispatchEnd中,读取目前AppMethodBeat中目前的index,记为end。这两者中间的数据则为这段时间内执行的方法的入栈、出栈信息。当这个Message执行时间超过指定的阈值(默认700ms)时,认为可能发生了慢方法,此时会进行进一步的分析。
至于doFrame中记录的数据,没有啥具体的用处,这是在最后打印了log而已。

代码和部分解释如下:

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/tracer/EvilMethodTracer.java

@Override
public void dispatchBegin(long beginMs, long cpuBeginMs, long token) {
    super.dispatchBegin(beginMs, cpuBeginMs, token);
    // 记录起始index
    indexRecord = AppMethodBeat.getInstance().maskIndex("EvilMethodTracer#dispatchBegin");
}

@Override
public void doFrame(String focusedActivityName, long start, long end, long frameCostMs, long inputCostNs, long animationCostNs, long traversalCostNs) {
    queueTypeCosts[0] = inputCostNs;
    queueTypeCosts[1] = animationCostNs;
    queueTypeCosts[2] = traversalCostNs;
}

@Override
public void dispatchEnd(long beginMs, long cpuBeginMs, long endMs, long cpuEndMs, long token, boolean isBelongFrame) {
    super.dispatchEnd(beginMs, cpuBeginMs, endMs, cpuEndMs, token, isBelongFrame);
    long start = config.isDevEnv() ? System.currentTimeMillis() : 0;
    try {
        // 若Message执行总耗时超过了阈值
        long dispatchCost = endMs - beginMs;
        if (dispatchCost >= evilThresholdMs) {
            // 则解析出这段时间内函数的调用堆栈
            long[] data = AppMethodBeat.getInstance().copyData(indexRecord);
            long[] queueCosts = new long[3];
            System.arraycopy(queueTypeCosts, 0, queueCosts, 0, 3);
            String scene = AppMethodBeat.getVisibleScene();
            // 在子线程进行分析上报
            MatrixHandlerThread.getDefaultHandler().post(new AnalyseTask(isForeground(), scene, data, queueCosts, cpuEndMs - cpuBeginMs, endMs - beginMs, endMs));
        }
    } finally {
        // 释放IndexRecord
        indexRecord.release();
        if (config.isDevEnv()) {
            String usage = Utils.calculateCpuUsage(cpuEndMs - cpuBeginMs, endMs - beginMs);
            MatrixLog.v(TAG, "[dispatchEnd] token:%s cost:%sms cpu:%sms usage:%s innerCost:%s",
                    token, endMs - beginMs, cpuEndMs - cpuBeginMs, usage, System.currentTimeMillis() - start);
        }
    }
}

下面我们看看AnalyseTask是如何分析并上报的:

private class AnalyseTask implements Runnable {
    ...

    void analyse() {

        // process
        // 获取进程的priority以及nice,原理是读取/proc/<pid>/stat中的数据
        int[] processStat = Utils.getProcessPriority(Process.myPid());
        // 计算出cpu使用率
        String usage = Utils.calculateCpuUsage(cpuCost, cost);
        LinkedList<MethodItem> stack = new LinkedList();
        if (data.length > 0) {
            // 将原始long[]数组合并成为MethodItem类型的List
            // 第3个参数(isStrict)与第4个参数(endTime)成对,若isStrict为true,则处理原始数据时会从第一个METHOD_ID_DISPATCH开始
            // 同时,若处理完毕后还有部分i数据没有匹配的o,则将endTime作为o的时间合并到List中
            TraceDataUtils.structuredDataToStack(data, stack, true, endMs);
            TraceDataUtils.trimStack(stack, Constants.TARGET_EVIL_METHOD_STACK, new TraceDataUtils.IStructuredDataFilter() {
                // 处理过程中需要被丢弃的条件
                @Override
                public boolean isFilter(long during, int filterCount) {
                    return during < filterCount * Constants.TIME_UPDATE_CYCLE_MS;
                }

                // 最多保留的item条数
                @Override
                public int getFilterMaxCount() {
                    return Constants.FILTER_STACK_MAX_COUNT;
                }

                // 最后处理完毕后,还是超过了maxCount,则调用此方法进行兜底
                @Override
                public void fallback(List<MethodItem> stack, int size) {
                    MatrixLog.w(TAG, "[fallback] size:%s targetSize:%s stack:%s", size, Constants.TARGET_EVIL_METHOD_STACK, stack);
                    Iterator iterator = stack.listIterator(Math.min(size, Constants.TARGET_EVIL_METHOD_STACK));
                    while (iterator.hasNext()) {
                        iterator.next();
                        iterator.remove();
                    }
                }
            });
        }


        StringBuilder reportBuilder = new StringBuilder();
        StringBuilder logcatBuilder = new StringBuilder();
        long stackCost = Math.max(cost, TraceDataUtils.stackToString(stack, reportBuilder, logcatBuilder));
        String stackKey = TraceDataUtils.getTreeKey(stack, stackCost);

        MatrixLog.w(TAG, "%s", printEvil(scene, processStat, isForeground, logcatBuilder, stack.size(), stackKey, usage, queueCost[0], queueCost[1], queueCost[2], cost)); // for logcat

        // report
        try {
            ...
        } catch (JSONException e) {
            MatrixLog.e(TAG, "[JSONException error: %s", e);
        }

    }

    @Override
    public void run() {
        analyse();
    }

    ...
}

AnalyseTask中比较耗脑的就是原始数据sBuffer如何进行整合以及裁剪,如何生成能够代表卡顿的key,方便进行聚合。这部分在Matrix-Wiki中也有一点介绍。摘抄如下:

堆栈聚类问题: 如果将收集的原始数据进行上报,数据量很大而且后台很难聚类有问题的堆栈,所以在上报之前需要对采集的数据进行简单的整合及裁剪,并分析出一个能代表卡顿堆栈的 key,方便后台聚合。

通过遍历采集的 buffer ,相邻 i 与 o 为一次完整函数执行,计算出一个调用树及每个函数执行耗时,并对每一级中的一些相同执行函数做聚合,最后通过一个简单策略,分析出主要耗时的那一级函数,作为代表卡顿堆栈的key。 matrix_stack

4.3 ANR监控AnrTracer

在ANR监控中,若发生了ANR,则需要解析这段时间内的调用堆栈。这部分逻辑与上面慢方法监控EvilMethodTracer基本一致。

下面考虑ANR如何进行判定。我们在Message执行开始时抛出一个定时任务,若该任务执行到了,则可以认为发生了ANR。若该任务在Message执行完毕之后被主动清除了,则说明没有ANR发生。这种思想与系统ANR的判定有相似之处。

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/tracer/AnrTracer.java

@Override
public void dispatchBegin(long beginMs, long cpuBeginMs, long token) {
    super.dispatchBegin(beginMs, cpuBeginMs, token);
    anrTask = new AnrHandleTask(AppMethodBeat.getInstance().maskIndex("AnrTracer#dispatchBegin"), token);
    if (traceConfig.isDevEnv()) {
        MatrixLog.v(TAG, "* [dispatchBegin] token:%s index:%s", token, anrTask.beginRecord.index);
    }
    // 抛炸弹
    anrHandler.postDelayed(anrTask, Constants.DEFAULT_ANR - (SystemClock.uptimeMillis() - token));
}

@Override
public void doFrame(String focusedActivityName, long start, long end, long frameCostMs, long inputCost, long animationCost, long traversalCost) {
    if (traceConfig.isDevEnv()) {
        MatrixLog.v(TAG, "--> [doFrame] activityName:%s frameCost:%sms [%s:%s:%s]ns", focusedActivityName, frameCostMs, inputCost, animationCost, traversalCost);
    }
}

@Override
public void dispatchEnd(long beginMs, long cpuBeginMs, long endMs, long cpuEndMs, long token, boolean isBelongFrame) {
    super.dispatchEnd(beginMs, cpuBeginMs, endMs, cpuEndMs, token, isBelongFrame);
    if (traceConfig.isDevEnv()) {
        MatrixLog.v(TAG, "[dispatchEnd] token:%s cost:%sms cpu:%sms usage:%s",
                token, endMs - beginMs, cpuEndMs - cpuBeginMs, Utils.calculateCpuUsage(cpuEndMs - cpuBeginMs, endMs - beginMs));
    }
    // 拆炸弹
    if (null != anrTask) {
        anrTask.getBeginRecord().release();
        anrHandler.removeCallbacks(anrTask);
    }
}

4.4 启动耗时StartUpTracer

启动监控涉及到的类有点多,但是思想不复杂。下面是StartUpTracer文件的注释,这里反映出了Matrix方案的启动耗时的思路:

firstMethod.i       LAUNCH_ACTIVITY   onWindowFocusChange   LAUNCH_ACTIVITY    onWindowFocusChange
^                         ^                   ^                     ^                  ^
|                         |                   |                     |                  |
|---------app---------|---|---firstActivity---|---------...---------|---careActivity---|
|<--applicationCost-->|
|<--------------firstScreenCost-------------->|
|<---------------------------------------coldCost------------------------------------->|
.                         |<-----warmCost---->|

简述如下:

之前插件在编译器为 Activity#onWindowFocusChanged 织入 AppMethodBeat.at 方法,这样可以获取每个 Activity 的 onWindowFocusChanged 回调时间。
然后在第一个 AppMethodBeat.i 方法调用时,记录此时的时间作为进程 zygote 后的时间;hook ActivityThread 中的 mH 中的 Callback ,通过检查第一个 Activity 或 Service 或 Receiver 的 what,以此时的时间作为 Application 创建结束时间,该时间与上面的时间之差记为 Application创建耗时
在第一个 Activity 的 onWindowFocusChange 回调时,此时的时间减去 zygote 时间即为 首屏启动耗时
第二个 Activity 的 onWindowFocusChange 回调时,时间减去 zygote 的时间即为 整个冷启动的时间

我们顺着上面的这个线理一下代码,首先是AppMethBeat.i方法里面的相关代码。

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/core/AppMethodBeat.java

public static void i(int methodId) {

    if (status <= STATUS_STOPPED) {
        return;
    }
    if (methodId >= METHOD_ID_MAX) {
        return;
    }

    if (status == STATUS_DEFAULT) {
        synchronized (statusLock) {
            if (status == STATUS_DEFAULT) {
                realExecute();
                status = STATUS_READY;
            }
        }
    }
    ...
}

status默认状态是STATUS_DEFAULT,因此第一次执行AppMethodBeat#i方法肯定会执行到realExecute()方法,这里面相当于就是初始化AppMethodBeat以及执行其他的一些操作了。

private static void realExecute() {
    MatrixLog.i(TAG, "[realExecute] timestamp:%s", System.currentTimeMillis());

    sCurrentDiffTime = SystemClock.uptimeMillis() - sDiffTime;

    sHandler.removeCallbacksAndMessages(null);
    sHandler.postDelayed(sUpdateDiffTimeRunnable, Constants.TIME_UPDATE_CYCLE_MS);
    sHandler.postDelayed(checkStartExpiredRunnable = new Runnable() {
        @Override
        public void run() {
            synchronized (statusLock) {
                MatrixLog.i(TAG, "[startExpired] timestamp:%s status:%s", System.currentTimeMillis(), status);
                if (status == STATUS_DEFAULT || status == STATUS_READY) {
                    status = STATUS_EXPIRED_START;
                }
            }
        }
    }, Constants.DEFAULT_RELEASE_BUFFER_DELAY);

    // hook mH,为StartUpTracer提供支持
    ActivityThreadHacker.hackSysHandlerCallback();
    LooperMonitor.register(looperMonitorListener);
}

我们关注一下ActivityThreadHacker.hackSysHandlerCallback()这个方法,这里面 hook 了 mH 的 mCallback,这样可以 Application 初始化结束的时间。

matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/hacker/ActivityThreadHacker.java

public class ActivityThreadHacker {
    private static final String TAG = "Matrix.ActivityThreadHacker";
    // 进程启动的时间
    private static long sApplicationCreateBeginTime = 0L;
    // 四大组件首次执行到的时间
    private static long sApplicationCreateEndTime = 0L;
    // Activity最后一次launch的时间
    private static long sLastLaunchActivityTime = 0L;
    public static AppMethodBeat.IndexRecord sLastLaunchActivityMethodIndex = new AppMethodBeat.IndexRecord();
    public static AppMethodBeat.IndexRecord sApplicationCreateBeginMethodIndex = new AppMethodBeat.IndexRecord();
    public static int sApplicationCreateScene = -100;

    public static void hackSysHandlerCallback() {
        try {
            sApplicationCreateBeginTime = SystemClock.uptimeMillis();
            sApplicationCreateBeginMethodIndex = AppMethodBeat.getInstance().maskIndex("ApplicationCreateBeginMethodIndex");
            Class<?> forName = Class.forName("android.app.ActivityThread");
            Field field = forName.getDeclaredField("sCurrentActivityThread");
            field.setAccessible(true);
            Object activityThreadValue = field.get(forName);
            Field mH = forName.getDeclaredField("mH");
            mH.setAccessible(true);
            Object handler = mH.get(activityThreadValue);
            Class<?> handlerClass = handler.getClass().getSuperclass();
            Field callbackField = handlerClass.getDeclaredField("mCallback");
            callbackField.setAccessible(true);
            Handler.Callback originalCallback = (Handler.Callback) callbackField.get(handler);
            HackCallback callback = new HackCallback(originalCallback);
            callbackField.set(handler, callback);
            MatrixLog.i(TAG, "hook system handler completed. start:%s SDK_INT:%s", sApplicationCreateBeginTime, Build.VERSION.SDK_INT);
        } catch (Exception e) {
            MatrixLog.e(TAG, "hook system handler err! %s", e.getCause().toString());
        }
    }

    public static long getApplicationCost() {
        return ActivityThreadHacker.sApplicationCreateEndTime - ActivityThreadHacker.sApplicationCreateBeginTime;
    }

    public static long getEggBrokenTime() {
        return ActivityThreadHacker.sApplicationCreateBeginTime;
    }

    public static long getLastLaunchActivityTime() {
        return ActivityThreadHacker.sLastLaunchActivityTime;
    }


    private final static class HackCallback implements Handler.Callback {
        private static final int LAUNCH_ACTIVITY = 100;
        private static final int CREATE_SERVICE = 114;
        private static final int RECEIVER = 113;
        public static final int EXECUTE_TRANSACTION = 159; // for Android 9.0
        private static boolean isCreated = false;
        private static int hasPrint = 10;

        private final Handler.Callback mOriginalCallback;

        HackCallback(Handler.Callback callback) {
            this.mOriginalCallback = callback;
        }

        @Override
        public boolean handleMessage(Message msg) {

            if (!AppMethodBeat.isRealTrace()) {
                return null != mOriginalCallback && mOriginalCallback.handleMessage(msg);
            }

            boolean isLaunchActivity = isLaunchActivity(msg);
            if (hasPrint > 0) {
                MatrixLog.i(TAG, "[handleMessage] msg.what:%s begin:%s isLaunchActivity:%s", msg.what, SystemClock.uptimeMillis(), isLaunchActivity);
                hasPrint--;
            }
            if (isLaunchActivity) {
                ActivityThreadHacker.sLastLaunchActivityTime = SystemClock.uptimeMillis();
                ActivityThreadHacker.sLastLaunchActivityMethodIndex = AppMethodBeat.getInstance().maskIndex("LastLaunchActivityMethodIndex");
            }

            if (!isCreated) {
                if (isLaunchActivity || msg.what == CREATE_SERVICE || msg.what == RECEIVER) { // todo for provider
                    ActivityThreadHacker.sApplicationCreateEndTime = SystemClock.uptimeMillis();
                    ActivityThreadHacker.sApplicationCreateScene = msg.what;
                    isCreated = true;
                }
            }

            return null != mOriginalCallback && mOriginalCallback.handleMessage(msg);
        }

        private Method method = null;

        private boolean isLaunchActivity(Message msg) {
            if (Build.VERSION.SDK_INT > Build.VERSION_CODES.O_MR1) {
                if (msg.what == EXECUTE_TRANSACTION && msg.obj != null) {
                    try {
                        if (null == method) {
                            Class clazz = Class.forName("android.app.servertransaction.ClientTransaction");
                            method = clazz.getDeclaredMethod("getCallbacks");
                            method.setAccessible(true);
                        }
                        List list = (List) method.invoke(msg.obj);
                        if (!list.isEmpty()) {
                            return list.get(0).getClass().getName().endsWith(".LaunchActivityItem");
                        }
                    } catch (Exception e) {
                        MatrixLog.e(TAG, "[isLaunchActivity] %s", e);
                    }
                }
                return msg.what == LAUNCH_ACTIVITY;
            } else {
                return msg.what == LAUNCH_ACTIVITY;
            }
        }
    }
}

上面这个类里面的代码比较简单,没有过度分析的必要。有关StartUpTracer相关的代码都用高亮标注了一下。在计算出Application初始化的起始时间点后,我们继续看看StartUpTracer#onActivityFocused里面的代码:

matrix/matrix/matrix-android/matrix-trace-canary/src/main/java/com/tencent/matrix/trace/tracer/StartupTracer.java

@Override
public void onActivityFocused(String activity) {
    // 若coldCost为初始值0,则说明这段代码从来没有运行过,那么认为是冷启动
    if (isColdStartup()) {
        // 若firstScreenCost为初始值0,则说明这是第一个获取焦点的Activity,记录时间差为首屏启动耗时
        if (firstScreenCost == 0) {
            this.firstScreenCost = uptimeMillis() - ActivityThreadHacker.getEggBrokenTime();
        }
        // 若已经展示过了首屏Activity,则此Activity是真正的MainActivity,记录此时时间差为冷启动耗时
        if (hasShowSplashActivity) {
            coldCost = uptimeMillis() - ActivityThreadHacker.getEggBrokenTime();
        } else {
            // 若还没有展示过首屏Activity
            if (splashActivities.contains(activity)) {
                // 且声明的首屏Activity列表中包含此Activity,则设置标志位
                hasShowSplashActivity = true;
            } else if (splashActivities.isEmpty()) {
                // 声明的首屏Activity列表为空,则整个冷启动耗时就为首屏启动耗时
                MatrixLog.i(TAG, "default splash activity[%s]", activity);
                coldCost = firstScreenCost;
            } else {
                MatrixLog.w(TAG, "pass this activity[%s] at duration of start up! splashActivities=%s", activity, splashActivities);
            }
        }
        // 分析冷启动耗时
        if (coldCost > 0) {
            analyse(ActivityThreadHacker.getApplicationCost(), firstScreenCost, coldCost, false);
        }

    } else if (isWarmStartUp()) {
        // 是否是温启动,这里isWarmStartUp标志位还依赖于监听ActivityLifecycleCallbacks,这里代码没有贴出来
        // 温启动时间是当前时间减去最后一次 launch Activity 的时间
        isWarmStartUp = false;
        long warmCost = uptimeMillis() - ActivityThreadHacker.getLastLaunchActivityTime();
        if (warmCost > 0) {
            analyse(ActivityThreadHacker.getApplicationCost(), firstScreenCost, warmCost, true);
        }
    }

}

private boolean isColdStartup() {
    return coldCost == 0;
}

private boolean isWarmStartUp() {
    return isWarmStartUp;
}

主要流程都在上面的代码中了,关键代码也有注释。至于analyse这个分析上报逻辑里面的部分,就是启动时间达到了阈值,则复制出函数调用堆栈并且分析的过程。这部分逻辑也与上面几个Tracer一直,这里不做过多分析。

Matrix方案的实用性

Matrix的方案适用于多Activity的架构,不适用于单Activity多Fragment的架构。对于后者,在使用上还需要一定的修改来进行适配。

至此,Matrix TraceCanary部分的原理已经分析完毕。还有一个AppMethodBeat相关的插桩插件,插桩插件代码量比较多。留到后面进行专门分析。

评论