1*6dbdd20aSAndroid Build Coastguard Worker// Copyright (C) 2024 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 {Trace} from '../../public/trace'; 16*6dbdd20aSAndroid Build Coastguard Workerimport {PerfettoPlugin} from '../../public/plugin'; 17*6dbdd20aSAndroid Build Coastguard Workerimport {CounterOptions} from '../../components/tracks/base_counter_track'; 18*6dbdd20aSAndroid Build Coastguard Workerimport {TrackNode} from '../../public/workspace'; 19*6dbdd20aSAndroid Build Coastguard Workerimport {createQueryCounterTrack} from '../../components/tracks/query_counter_track'; 20*6dbdd20aSAndroid Build Coastguard Worker 21*6dbdd20aSAndroid Build Coastguard Workerexport default class implements PerfettoPlugin { 22*6dbdd20aSAndroid Build Coastguard Worker static readonly id = 'dev.perfetto.CpuidleTimeInState'; 23*6dbdd20aSAndroid Build Coastguard Worker private async addCounterTrack( 24*6dbdd20aSAndroid Build Coastguard Worker ctx: Trace, 25*6dbdd20aSAndroid Build Coastguard Worker name: string, 26*6dbdd20aSAndroid Build Coastguard Worker query: string, 27*6dbdd20aSAndroid Build Coastguard Worker group?: TrackNode, 28*6dbdd20aSAndroid Build Coastguard Worker options?: Partial<CounterOptions>, 29*6dbdd20aSAndroid Build Coastguard Worker ) { 30*6dbdd20aSAndroid Build Coastguard Worker const uri = `/cpuidle_time_in_state_${name}`; 31*6dbdd20aSAndroid Build Coastguard Worker const track = await createQueryCounterTrack({ 32*6dbdd20aSAndroid Build Coastguard Worker trace: ctx, 33*6dbdd20aSAndroid Build Coastguard Worker uri, 34*6dbdd20aSAndroid Build Coastguard Worker data: { 35*6dbdd20aSAndroid Build Coastguard Worker sqlSource: query, 36*6dbdd20aSAndroid Build Coastguard Worker columns: ['ts', 'value'], 37*6dbdd20aSAndroid Build Coastguard Worker }, 38*6dbdd20aSAndroid Build Coastguard Worker columns: {ts: 'ts', value: 'value'}, 39*6dbdd20aSAndroid Build Coastguard Worker options, 40*6dbdd20aSAndroid Build Coastguard Worker }); 41*6dbdd20aSAndroid Build Coastguard Worker ctx.tracks.registerTrack({ 42*6dbdd20aSAndroid Build Coastguard Worker uri, 43*6dbdd20aSAndroid Build Coastguard Worker title: name, 44*6dbdd20aSAndroid Build Coastguard Worker track, 45*6dbdd20aSAndroid Build Coastguard Worker }); 46*6dbdd20aSAndroid Build Coastguard Worker const trackNode = new TrackNode({uri, title: name}); 47*6dbdd20aSAndroid Build Coastguard Worker if (group) { 48*6dbdd20aSAndroid Build Coastguard Worker group.addChildInOrder(trackNode); 49*6dbdd20aSAndroid Build Coastguard Worker } 50*6dbdd20aSAndroid Build Coastguard Worker } 51*6dbdd20aSAndroid Build Coastguard Worker 52*6dbdd20aSAndroid Build Coastguard Worker async onTraceLoad(ctx: Trace): Promise<void> { 53*6dbdd20aSAndroid Build Coastguard Worker const group = new TrackNode({ 54*6dbdd20aSAndroid Build Coastguard Worker title: 'Cpuidle Time In State', 55*6dbdd20aSAndroid Build Coastguard Worker isSummary: true, 56*6dbdd20aSAndroid Build Coastguard Worker }); 57*6dbdd20aSAndroid Build Coastguard Worker 58*6dbdd20aSAndroid Build Coastguard Worker const e = ctx.engine; 59*6dbdd20aSAndroid Build Coastguard Worker await e.query(`INCLUDE PERFETTO MODULE linux.cpu.idle_time_in_state;`); 60*6dbdd20aSAndroid Build Coastguard Worker const result = await e.query( 61*6dbdd20aSAndroid Build Coastguard Worker `select distinct state_name from cpu_idle_time_in_state_counters`, 62*6dbdd20aSAndroid Build Coastguard Worker ); 63*6dbdd20aSAndroid Build Coastguard Worker const it = result.iter({state_name: 'str'}); 64*6dbdd20aSAndroid Build Coastguard Worker for (; it.valid(); it.next()) { 65*6dbdd20aSAndroid Build Coastguard Worker this.addCounterTrack( 66*6dbdd20aSAndroid Build Coastguard Worker ctx, 67*6dbdd20aSAndroid Build Coastguard Worker it.state_name, 68*6dbdd20aSAndroid Build Coastguard Worker ` 69*6dbdd20aSAndroid Build Coastguard Worker select 70*6dbdd20aSAndroid Build Coastguard Worker ts, 71*6dbdd20aSAndroid Build Coastguard Worker idle_percentage as value 72*6dbdd20aSAndroid Build Coastguard Worker from cpu_idle_time_in_state_counters 73*6dbdd20aSAndroid Build Coastguard Worker where state_name = '${it.state_name}' 74*6dbdd20aSAndroid Build Coastguard Worker `, 75*6dbdd20aSAndroid Build Coastguard Worker group, 76*6dbdd20aSAndroid Build Coastguard Worker {unit: 'percent'}, 77*6dbdd20aSAndroid Build Coastguard Worker ); 78*6dbdd20aSAndroid Build Coastguard Worker } 79*6dbdd20aSAndroid Build Coastguard Worker if (group.hasChildren) { 80*6dbdd20aSAndroid Build Coastguard Worker ctx.workspace.addChildInOrder(group); 81*6dbdd20aSAndroid Build Coastguard Worker } 82*6dbdd20aSAndroid Build Coastguard Worker } 83*6dbdd20aSAndroid Build Coastguard Worker} 84