1*6dbdd20aSAndroid Build Coastguard Worker// Copyright (C) 2021 The Android Open Source Project 2*6dbdd20aSAndroid Build Coastguard Worker// 3*6dbdd20aSAndroid Build Coastguard Worker// Licensed under the Apache License, Version 2.0 (the "License"); 4*6dbdd20aSAndroid Build Coastguard Worker// you may not use this file except in compliance with the License. 5*6dbdd20aSAndroid Build Coastguard Worker// You may obtain a copy of the License at 6*6dbdd20aSAndroid Build Coastguard Worker// 7*6dbdd20aSAndroid Build Coastguard Worker// http://www.apache.org/licenses/LICENSE-2.0 8*6dbdd20aSAndroid Build Coastguard Worker// 9*6dbdd20aSAndroid Build Coastguard Worker// Unless required by applicable law or agreed to in writing, software 10*6dbdd20aSAndroid Build Coastguard Worker// distributed under the License is distributed on an "AS IS" BASIS, 11*6dbdd20aSAndroid Build Coastguard Worker// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12*6dbdd20aSAndroid Build Coastguard Worker// See the License for the specific language governing permissions and 13*6dbdd20aSAndroid Build Coastguard Worker// limitations under the License. 14*6dbdd20aSAndroid Build Coastguard Worker 15*6dbdd20aSAndroid Build Coastguard Workerimport {TrackData} from '../../components/tracks/track_data'; 16*6dbdd20aSAndroid Build Coastguard Workerimport {PERF_SAMPLES_PROFILE_TRACK_KIND} from '../../public/track_kinds'; 17*6dbdd20aSAndroid Build Coastguard Workerimport {Trace} from '../../public/trace'; 18*6dbdd20aSAndroid Build Coastguard Workerimport {PerfettoPlugin} from '../../public/plugin'; 19*6dbdd20aSAndroid Build Coastguard Workerimport {NUM, NUM_NULL, STR_NULL} from '../../trace_processor/query_result'; 20*6dbdd20aSAndroid Build Coastguard Workerimport {assertExists} from '../../base/logging'; 21*6dbdd20aSAndroid Build Coastguard Workerimport { 22*6dbdd20aSAndroid Build Coastguard Worker ProcessPerfSamplesProfileTrack, 23*6dbdd20aSAndroid Build Coastguard Worker ThreadPerfSamplesProfileTrack, 24*6dbdd20aSAndroid Build Coastguard Worker} from './perf_samples_profile_track'; 25*6dbdd20aSAndroid Build Coastguard Workerimport {getThreadUriPrefix} from '../../public/utils'; 26*6dbdd20aSAndroid Build Coastguard Workerimport {TrackNode} from '../../public/workspace'; 27*6dbdd20aSAndroid Build Coastguard Workerimport ProcessThreadGroupsPlugin from '../dev.perfetto.ProcessThreadGroups'; 28*6dbdd20aSAndroid Build Coastguard Worker 29*6dbdd20aSAndroid Build Coastguard Workerexport interface Data extends TrackData { 30*6dbdd20aSAndroid Build Coastguard Worker tsStarts: BigInt64Array; 31*6dbdd20aSAndroid Build Coastguard Worker} 32*6dbdd20aSAndroid Build Coastguard Worker 33*6dbdd20aSAndroid Build Coastguard Workerfunction makeUriForProc(upid: number) { 34*6dbdd20aSAndroid Build Coastguard Worker return `/process_${upid}/perf_samples_profile`; 35*6dbdd20aSAndroid Build Coastguard Worker} 36*6dbdd20aSAndroid Build Coastguard Worker 37*6dbdd20aSAndroid Build Coastguard Workerexport default class implements PerfettoPlugin { 38*6dbdd20aSAndroid Build Coastguard Worker static readonly id = 'dev.perfetto.PerfSamplesProfile'; 39*6dbdd20aSAndroid Build Coastguard Worker static readonly dependencies = [ProcessThreadGroupsPlugin]; 40*6dbdd20aSAndroid Build Coastguard Worker 41*6dbdd20aSAndroid Build Coastguard Worker async onTraceLoad(ctx: Trace): Promise<void> { 42*6dbdd20aSAndroid Build Coastguard Worker const pResult = await ctx.engine.query(` 43*6dbdd20aSAndroid Build Coastguard Worker select distinct upid 44*6dbdd20aSAndroid Build Coastguard Worker from perf_sample 45*6dbdd20aSAndroid Build Coastguard Worker join thread using (utid) 46*6dbdd20aSAndroid Build Coastguard Worker where callsite_id is not null and upid is not null 47*6dbdd20aSAndroid Build Coastguard Worker `); 48*6dbdd20aSAndroid Build Coastguard Worker for (const it = pResult.iter({upid: NUM}); it.valid(); it.next()) { 49*6dbdd20aSAndroid Build Coastguard Worker const upid = it.upid; 50*6dbdd20aSAndroid Build Coastguard Worker const uri = makeUriForProc(upid); 51*6dbdd20aSAndroid Build Coastguard Worker const title = `Process Callstacks`; 52*6dbdd20aSAndroid Build Coastguard Worker ctx.tracks.registerTrack({ 53*6dbdd20aSAndroid Build Coastguard Worker uri, 54*6dbdd20aSAndroid Build Coastguard Worker title, 55*6dbdd20aSAndroid Build Coastguard Worker tags: { 56*6dbdd20aSAndroid Build Coastguard Worker kind: PERF_SAMPLES_PROFILE_TRACK_KIND, 57*6dbdd20aSAndroid Build Coastguard Worker upid, 58*6dbdd20aSAndroid Build Coastguard Worker }, 59*6dbdd20aSAndroid Build Coastguard Worker track: new ProcessPerfSamplesProfileTrack(ctx, uri, upid), 60*6dbdd20aSAndroid Build Coastguard Worker }); 61*6dbdd20aSAndroid Build Coastguard Worker const group = ctx.plugins 62*6dbdd20aSAndroid Build Coastguard Worker .getPlugin(ProcessThreadGroupsPlugin) 63*6dbdd20aSAndroid Build Coastguard Worker .getGroupForProcess(upid); 64*6dbdd20aSAndroid Build Coastguard Worker const track = new TrackNode({uri, title, sortOrder: -40}); 65*6dbdd20aSAndroid Build Coastguard Worker group?.addChildInOrder(track); 66*6dbdd20aSAndroid Build Coastguard Worker } 67*6dbdd20aSAndroid Build Coastguard Worker const tResult = await ctx.engine.query(` 68*6dbdd20aSAndroid Build Coastguard Worker select distinct 69*6dbdd20aSAndroid Build Coastguard Worker utid, 70*6dbdd20aSAndroid Build Coastguard Worker tid, 71*6dbdd20aSAndroid Build Coastguard Worker thread.name as threadName, 72*6dbdd20aSAndroid Build Coastguard Worker upid 73*6dbdd20aSAndroid Build Coastguard Worker from perf_sample 74*6dbdd20aSAndroid Build Coastguard Worker join thread using (utid) 75*6dbdd20aSAndroid Build Coastguard Worker where callsite_id is not null 76*6dbdd20aSAndroid Build Coastguard Worker `); 77*6dbdd20aSAndroid Build Coastguard Worker for ( 78*6dbdd20aSAndroid Build Coastguard Worker const it = tResult.iter({ 79*6dbdd20aSAndroid Build Coastguard Worker utid: NUM, 80*6dbdd20aSAndroid Build Coastguard Worker tid: NUM, 81*6dbdd20aSAndroid Build Coastguard Worker threadName: STR_NULL, 82*6dbdd20aSAndroid Build Coastguard Worker upid: NUM_NULL, 83*6dbdd20aSAndroid Build Coastguard Worker }); 84*6dbdd20aSAndroid Build Coastguard Worker it.valid(); 85*6dbdd20aSAndroid Build Coastguard Worker it.next() 86*6dbdd20aSAndroid Build Coastguard Worker ) { 87*6dbdd20aSAndroid Build Coastguard Worker const {threadName, utid, tid, upid} = it; 88*6dbdd20aSAndroid Build Coastguard Worker const title = 89*6dbdd20aSAndroid Build Coastguard Worker threadName === null 90*6dbdd20aSAndroid Build Coastguard Worker ? `Thread Callstacks ${tid}` 91*6dbdd20aSAndroid Build Coastguard Worker : `${threadName} Callstacks ${tid}`; 92*6dbdd20aSAndroid Build Coastguard Worker const uri = `${getThreadUriPrefix(upid, utid)}_perf_samples_profile`; 93*6dbdd20aSAndroid Build Coastguard Worker ctx.tracks.registerTrack({ 94*6dbdd20aSAndroid Build Coastguard Worker uri, 95*6dbdd20aSAndroid Build Coastguard Worker title, 96*6dbdd20aSAndroid Build Coastguard Worker tags: { 97*6dbdd20aSAndroid Build Coastguard Worker kind: PERF_SAMPLES_PROFILE_TRACK_KIND, 98*6dbdd20aSAndroid Build Coastguard Worker utid, 99*6dbdd20aSAndroid Build Coastguard Worker upid: upid ?? undefined, 100*6dbdd20aSAndroid Build Coastguard Worker }, 101*6dbdd20aSAndroid Build Coastguard Worker track: new ThreadPerfSamplesProfileTrack(ctx, uri, utid), 102*6dbdd20aSAndroid Build Coastguard Worker }); 103*6dbdd20aSAndroid Build Coastguard Worker const group = ctx.plugins 104*6dbdd20aSAndroid Build Coastguard Worker .getPlugin(ProcessThreadGroupsPlugin) 105*6dbdd20aSAndroid Build Coastguard Worker .getGroupForThread(utid); 106*6dbdd20aSAndroid Build Coastguard Worker const track = new TrackNode({uri, title, sortOrder: -50}); 107*6dbdd20aSAndroid Build Coastguard Worker group?.addChildInOrder(track); 108*6dbdd20aSAndroid Build Coastguard Worker } 109*6dbdd20aSAndroid Build Coastguard Worker 110*6dbdd20aSAndroid Build Coastguard Worker ctx.onTraceReady.addListener(async () => { 111*6dbdd20aSAndroid Build Coastguard Worker await selectPerfSample(ctx); 112*6dbdd20aSAndroid Build Coastguard Worker }); 113*6dbdd20aSAndroid Build Coastguard Worker } 114*6dbdd20aSAndroid Build Coastguard Worker} 115*6dbdd20aSAndroid Build Coastguard Worker 116*6dbdd20aSAndroid Build Coastguard Workerasync function selectPerfSample(ctx: Trace) { 117*6dbdd20aSAndroid Build Coastguard Worker const profile = await assertExists(ctx.engine).query(` 118*6dbdd20aSAndroid Build Coastguard Worker select upid 119*6dbdd20aSAndroid Build Coastguard Worker from perf_sample 120*6dbdd20aSAndroid Build Coastguard Worker join thread using (utid) 121*6dbdd20aSAndroid Build Coastguard Worker where callsite_id is not null 122*6dbdd20aSAndroid Build Coastguard Worker order by ts desc 123*6dbdd20aSAndroid Build Coastguard Worker limit 1 124*6dbdd20aSAndroid Build Coastguard Worker `); 125*6dbdd20aSAndroid Build Coastguard Worker if (profile.numRows() !== 1) return; 126*6dbdd20aSAndroid Build Coastguard Worker const row = profile.firstRow({upid: NUM}); 127*6dbdd20aSAndroid Build Coastguard Worker const upid = row.upid; 128*6dbdd20aSAndroid Build Coastguard Worker 129*6dbdd20aSAndroid Build Coastguard Worker // Create an area selection over the first process with a perf samples track 130*6dbdd20aSAndroid Build Coastguard Worker ctx.selection.selectArea({ 131*6dbdd20aSAndroid Build Coastguard Worker start: ctx.traceInfo.start, 132*6dbdd20aSAndroid Build Coastguard Worker end: ctx.traceInfo.end, 133*6dbdd20aSAndroid Build Coastguard Worker trackUris: [makeUriForProc(upid)], 134*6dbdd20aSAndroid Build Coastguard Worker }); 135*6dbdd20aSAndroid Build Coastguard Worker} 136