// 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 {Trace} from '../../public/trace'; import {PerfettoPlugin} from '../../public/plugin'; import {CounterOptions} from '../../components/tracks/base_counter_track'; import {TrackNode} from '../../public/workspace'; import {createQueryCounterTrack} from '../../components/tracks/query_counter_track'; import StandardGroupsPlugin from '../dev.perfetto.StandardGroups'; import {NUM, STR} from '../../trace_processor/query_result'; export default class implements PerfettoPlugin { static readonly id = 'dev.perfetto.CpuidleTimeInState'; static readonly dependencies = [StandardGroupsPlugin]; private async addCounterTrack( ctx: Trace, name: string, query: string, group: TrackNode, options?: Partial, ) { const uri = `/cpuidle_time_in_state_${name}`; const track = await createQueryCounterTrack({ trace: ctx, uri, data: { sqlSource: query, columns: ['ts', 'value'], }, columns: {ts: 'ts', value: 'value'}, options, }); ctx.tracks.registerTrack({ uri, title: name, track, }); const node = new TrackNode({uri, title: name}); group.addChildInOrder(node); } async addIdleStateTrack( ctx: Trace, state: string, group: TrackNode, ): Promise { await this.addCounterTrack( ctx, `cpuidle.${state}`, ` select ts, idle_percentage as value from linux_cpu_idle_time_in_state_counters where state = '${state}' `, group, {unit: 'percent', yOverrideMaximum: 100, yOverrideMinimum: 0}, ); } async addPerCpuIdleStateTrack( ctx: Trace, state: string, cpu: number, group: TrackNode, ): Promise { await this.addCounterTrack( ctx, `cpuidle.cpu${cpu}.${state} Residency`, ` select ts, idle_percentage as value from linux_per_cpu_idle_time_in_state_counters where state = '${state}' AND cpu = ${cpu} `, group, {unit: 'percent', yOverrideMaximum: 100, yOverrideMinimum: 0}, ); } async onTraceLoad(ctx: Trace): Promise { const group = new TrackNode({ title: 'CPU Idle Time In State', isSummary: true, }); const e = ctx.engine; await e.query(`INCLUDE PERFETTO MODULE linux.cpu.idle_time_in_state;`); const states = await e.query( `select distinct state from linux_cpu_idle_time_in_state_counters`, ); const it = states.iter({state: STR}); for (; it.valid(); it.next()) { await this.addIdleStateTrack(ctx, it.state, group); } if (group.hasChildren) { const cpuGroup = ctx.plugins .getPlugin(StandardGroupsPlugin) .getOrCreateStandardGroup(ctx.workspace, 'CPU'); cpuGroup.addChildInOrder(group); } const perCpuGroup = new TrackNode({ title: 'CPU Idle Per Cpu Time In State', isSummary: true, }); const perCpuStates = await e.query( `select distinct state, cpu from linux_per_cpu_idle_time_in_state_counters`, ); const pIt = perCpuStates.iter({state: STR, cpu: NUM}); for (; pIt.valid(); pIt.next()) { await this.addPerCpuIdleStateTrack(ctx, pIt.state, pIt.cpu, perCpuGroup); } if (perCpuGroup.hasChildren) { const cpuGroup = ctx.plugins .getPlugin(StandardGroupsPlugin) .getOrCreateStandardGroup(ctx.workspace, 'CPU'); cpuGroup.addChildInOrder(perCpuGroup); } } }