xref: /aosp_15_r20/external/perfetto/ui/src/plugins/dev.perfetto.PerfSamplesProfile/index.ts (revision 6dbdd20afdafa5e3ca9b8809fa73465d530080dc)
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