blob: e69d6556a11b3b8856d5108d7ae8b84407f3b414 [file] [edit]
// Copyright (C) 2023 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 m from 'mithril';
import {PerfettoPlugin} from '../../public/plugin';
import {Trace} from '../../public/trace';
import {TrackNode} from '../../public/workspace';
import {NUM} from '../../trace_processor/query_result';
import {Cpu} from '../../components/cpu';
import {FtraceFilter, FtracePluginState as FtraceFilters} from './common';
import {FtraceExplorer, FtraceExplorerCache} from './ftrace_explorer';
import {createFtraceTrack} from './ftrace_track';
const VERSION = 1;
const DEFAULT_STATE: FtraceFilters = {
version: VERSION,
filter: {
excludeList: [],
},
};
export default class implements PerfettoPlugin {
static readonly id = 'dev.perfetto.Ftrace';
async onTraceLoad(ctx: Trace): Promise<void> {
const store = ctx.mountStore<FtraceFilters>(
'dev.perfetto.FtraceFilters',
(init: unknown) => {
if (
typeof init === 'object' &&
init !== null &&
'version' in init &&
init.version === VERSION
) {
return init as {} as FtraceFilters;
} else {
return DEFAULT_STATE;
}
},
);
const filterStore = store.createSubStore(
['filter'],
(x) => x as FtraceFilter,
);
const ftraceTabUri = 'perfetto.FtraceRaw#FtraceEventsTab';
let hasExpandedOnce = false;
const cpus = await getFtraceCpus(ctx);
const group = new TrackNode({
name: 'Ftrace Events',
sortOrder: -5,
isSummary: true,
onExpand: () => {
if (!hasExpandedOnce) {
hasExpandedOnce = true;
ctx.tabs.showTab(ftraceTabUri);
}
},
});
for (const cpu of cpus) {
const uri = `/ftrace/cpu${cpu.ucpu}`;
ctx.tracks.registerTrack({
uri,
description: `Ftrace events for CPU ${cpu.toString()}`,
tags: {
cpu: cpu.cpu,
},
renderer: createFtraceTrack(ctx, uri, cpu.ucpu, filterStore),
});
const track = new TrackNode({
uri,
name: `Ftrace Track for CPU ${cpu.toString()}`,
});
group.addChildInOrder(track);
}
if (group.children.length) {
ctx.defaultWorkspace.addChildInOrder(group);
}
const cache: FtraceExplorerCache = {
state: 'blank',
counters: [],
};
ctx.tabs.registerTab({
uri: ftraceTabUri,
isEphemeral: false,
content: {
render: () =>
m(FtraceExplorer, {
filterStore,
cache,
trace: ctx,
}),
getTitle: () => 'Ftrace Events',
},
});
ctx.commands.registerCommand({
id: 'dev.perfetto.ShowFtraceTab',
name: 'Show ftrace tab',
callback: () => {
ctx.tabs.showTab(ftraceTabUri);
},
});
}
}
/**
* Get the list of unique cpus in the ftrace_event table.
*/
async function getFtraceCpus(ctx: Trace): Promise<Cpu[]> {
const queryRes = await ctx.engine.query(`
SELECT DISTINCT
ucpu,
cpu.machine_id AS machine_id,
cpu.cpu AS cpu
FROM ftrace_event
JOIN cpu USING (ucpu)
ORDER BY ucpu
`);
const ucpus: Cpu[] = [];
for (
const it = queryRes.iter({ucpu: NUM, machine_id: NUM, cpu: NUM});
it.valid();
it.next()
) {
ucpus.push(new Cpu(it.ucpu, it.cpu, it.machine_id));
}
return ucpus;
}