blob: 191da87caf40608c1909f34199b457d550f5db2e [file] [log] [blame] [edit]
// 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<CounterOptions>,
) {
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<void> {
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<void> {
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<void> {
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);
}
}
}