| // Copyright (C) 2024 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 {assertExists} from '../base/logging'; |
| import {uuidv4} from '../base/uuid'; |
| import {Actions, AddTrackArgs} from '../common/actions'; |
| import {InThreadTrackSortKey} from '../common/state'; |
| import {TrackDescriptor} from '../public/tracks'; |
| import {Engine} from '../trace_processor/engine'; |
| import {NUM} from '../trace_processor/query_result'; |
| import {globals} from './globals'; |
| import {VisualisedArgsTrack} from './visualized_args_track'; |
| |
| const VISUALISED_ARGS_SLICE_TRACK_URI_PREFIX = 'perfetto.VisualisedArgs'; |
| |
| // We need to add tracks from the core and from plugins. In order to add a debug |
| // track we need to pass a context through with we can add the track. This is |
| // different for plugins vs the core. This interface defines the generic shape |
| // of this context, which can be supplied from a plugin or built from globals. |
| // |
| // TODO(stevegolton): In the future, both the core and plugins should have |
| // access to some Context object which implements the various things we want to |
| // do in a generic way, so that we don't have to do this mangling to get this to |
| // work. |
| interface Context { |
| engine: Engine; |
| registerTrack(track: TrackDescriptor): unknown; |
| } |
| |
| export async function addVisualisedArgTracks(ctx: Context, argName: string) { |
| const escapedArgName = argName.replace(/[^a-zA-Z]/g, '_'); |
| const tableName = `__arg_visualisation_helper_${escapedArgName}_slice`; |
| |
| const result = await ctx.engine.query(` |
| drop table if exists ${tableName}; |
| |
| create table ${tableName} as |
| with slice_with_arg as ( |
| select |
| slice.id, |
| slice.track_id, |
| slice.ts, |
| slice.dur, |
| slice.thread_dur, |
| NULL as cat, |
| args.display_value as name |
| from slice |
| join args using (arg_set_id) |
| where args.key='${argName}' |
| ) |
| select |
| *, |
| (select count() |
| from ancestor_slice(s1.id) s2 |
| join slice_with_arg s3 on s2.id=s3.id |
| ) as depth |
| from slice_with_arg s1 |
| order by id; |
| |
| select |
| track_id as trackId, |
| max(depth) as maxDepth |
| from ${tableName} |
| group by track_id; |
| `); |
| |
| const tracksToAdd: AddTrackArgs[] = []; |
| const it = result.iter({trackId: NUM, maxDepth: NUM}); |
| const addedTrackKeys: string[] = []; |
| for (; it.valid(); it.next()) { |
| const trackId = it.trackId; |
| const maxDepth = it.maxDepth; |
| const trackKey = globals.trackManager.trackKeyByTrackId.get(trackId); |
| const track = globals.state.tracks[assertExists(trackKey)]; |
| const utid = (track.trackSortKey as {utid?: number}).utid; |
| const key = uuidv4(); |
| addedTrackKeys.push(key); |
| |
| const uri = `${VISUALISED_ARGS_SLICE_TRACK_URI_PREFIX}#${uuidv4()}`; |
| ctx.registerTrack({ |
| uri, |
| title: argName, |
| chips: ['metric'], |
| trackFactory: (trackCtx) => { |
| return new VisualisedArgsTrack({ |
| engine: ctx.engine, |
| trackKey: trackCtx.trackKey, |
| trackId, |
| maxDepth, |
| argName, |
| }); |
| }, |
| }); |
| |
| tracksToAdd.push({ |
| key, |
| trackGroup: track.trackGroup, |
| name: argName, |
| trackSortKey: |
| utid === undefined |
| ? track.trackSortKey |
| : {utid, priority: InThreadTrackSortKey.VISUALISED_ARGS_TRACK}, |
| uri, |
| }); |
| } |
| |
| globals.dispatchMultiple([ |
| Actions.addTracks({tracks: tracksToAdd}), |
| Actions.sortThreadTracks({}), |
| ]); |
| } |