blob: a5217249ea72e4c5c9a66823c7819518e9d4a65e [file] [log] [blame]
// Copyright (C) 2023 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import {
Plugin,
PluginContext,
PluginContextTrace,
PluginDescriptor,
} from '../../public';
class AndroidPerf implements Plugin {
onActivate(_ctx: PluginContext): void {}
async onTraceLoad(ctx: PluginContextTrace): Promise<void> {
ctx.registerCommand({
id: 'dev.perfetto.AndroidPerf#BinderSystemServerIncoming',
name: 'Run query: system_server incoming binder graph',
callback: () => ctx.tabs.openQuery(
`INCLUDE PERFETTO MODULE android.binder;
SELECT * FROM android_binder_incoming_graph((SELECT upid FROM process WHERE name = 'system_server'))`,
'system_server incoming binder graph'),
});
ctx.registerCommand({
id: 'dev.perfetto.AndroidPerf#BinderSystemServerOutgoing',
name: 'Run query: system_server outgoing binder graph',
callback: () => ctx.tabs.openQuery(
`INCLUDE PERFETTO MODULE android.binder;
SELECT * FROM android_binder_outgoing_graph((SELECT upid FROM process WHERE name = 'system_server'))`,
'system_server outgoing binder graph'),
});
ctx.registerCommand({
id: 'dev.perfetto.AndroidPerf#MonitorContentionSystemServer',
name: 'Run query: system_server monitor_contention graph',
callback: () => ctx.tabs.openQuery(
`INCLUDE PERFETTO MODULE android.monitor_contention;
SELECT * FROM android_monitor_contention_graph((SELECT upid FROM process WHERE name = 'system_server'))`,
'system_server monitor_contention graph'),
});
ctx.registerCommand({
id: 'dev.perfetto.AndroidPerf#BinderAll',
name: 'Run query: all process binder graph',
callback: () => ctx.tabs.openQuery(
`INCLUDE PERFETTO MODULE android.binder;
SELECT * FROM android_binder_graph(-1000, 1000, -1000, 1000)`,
'all process binder graph'),
});
ctx.registerCommand({
id: 'dev.perfetto.AndroidPerf#ThreadClusterDistribution',
name: 'Run query: runtime cluster distribution for a thread',
callback: async (tid) => {
if (tid === undefined) {
tid = prompt('Enter a thread tid', '');
if (tid === null) return;
}
ctx.tabs.openQuery(`
INCLUDE PERFETTO MODULE common.cpus;
WITH
total_runtime AS (
SELECT sum(dur) AS total_runtime
FROM sched s
LEFT JOIN thread t
USING (utid)
WHERE t.tid = ${tid}
)
SELECT
c.size AS cluster,
sum(dur)/1e6 AS total_dur_ms,
sum(dur) * 1.0 / (SELECT * FROM total_runtime) AS percentage
FROM sched s
LEFT JOIN thread t
USING (utid)
LEFT JOIN cpus c
ON s.cpu = c.cpu_index
WHERE t.tid = ${tid}
GROUP BY 1`, `runtime cluster distrubtion for tid ${tid}`);
},
});
ctx.registerCommand({
id: 'dev.perfetto.AndroidPerf#SchedLatency',
name: 'Run query: top 50 sched latency for a thread',
callback: async (tid) => {
if (tid === undefined) {
tid = prompt('Enter a thread tid', '');
if (tid === null) return;
}
ctx.tabs.openQuery(`
SELECT ts.*, t.tid, t.name, tt.id AS track_id
FROM thread_state ts
LEFT JOIN thread_track tt
USING (utid)
LEFT JOIN thread t
USING (utid)
WHERE ts.state IN ('R', 'R+') AND tid = ${tid}
ORDER BY dur DESC
LIMIT 50`, `top 50 sched latency slice for tid ${tid}`);
},
});
}
}
export const plugin: PluginDescriptor = {
pluginId: 'dev.perfetto.AndroidPerf',
plugin: AndroidPerf,
};