blob: 3cecdcecf77bc9afde98ffb85bedd17bc81f2630 [file]
// 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 m from 'mithril';
import {exists} from '../../base/utils';
import type {ColumnDef} from '../../components/aggregation';
import type {Aggregator} from '../../components/aggregation_adapter';
import type {AreaSelection} from '../../public/selection';
import {CPU_SLICE_TRACK_KIND} from '../../public/track_kinds';
import type {Engine} from '../../trace_processor/engine';
import type {SqlValue} from '../../trace_processor/query_result';
import {RadioGroup} from '../../widgets/radio_group';
export class WattsonProcessSelectionAggregator implements Aggregator {
readonly id = 'wattson_plugin_process_aggregation';
private scaleNumericData: boolean = false;
probe(area: AreaSelection) {
const selectedCpus: number[] = [];
for (const trackInfo of area.tracks) {
trackInfo?.tags?.kinds?.includes(CPU_SLICE_TRACK_KIND) &&
exists(trackInfo.tags.cpu) &&
selectedCpus.push(trackInfo.tags.cpu);
}
if (selectedCpus.length === 0) return undefined;
return {
prepareData: async (engine: Engine) => {
await engine.query(`drop view if exists ${this.id};`);
// Prerequisite tables are already generated by Wattson thread aggregation,
// which is run prior to execution of this module
await engine.query(`
CREATE PERFETTO VIEW ${this.id} AS
WITH base AS (
SELECT
ROUND(SUM(estimated_mw), 3) as active_mw,
ROUND(SUM(estimated_mws), 3) as active_mws,
ROUND(SUM(idle_transitions_mws), 3) as idle_cost_mws,
ROUND(SUM(total_mws), 3) as total_mws,
pid,
process_name,
upid
FROM wattson_plugin_thread_summary
GROUP BY upid
)
SELECT
*,
total_mws / (SUM(total_mws) OVER()) AS percent_of_total_energy
FROM base;
`);
return {
tableName: this.id,
};
},
};
}
renderTopbarControls(): m.Children {
return m(
RadioGroup,
{
selectedValue: this.scaleNumericData ? 'uw' : 'mw',
onValueChange: (value) => {
this.scaleNumericData = value === 'uw';
},
title: 'Select power units',
},
[
m(RadioGroup.Button, {value: 'uw'}, 'µW'),
m(RadioGroup.Button, {value: 'mw'}, 'mW'),
],
);
}
private powerUnits(): string {
return this.scaleNumericData ? 'µW' : 'mW';
}
private renderMilliwatts(value: SqlValue): m.Children {
if (this.scaleNumericData && typeof value === 'number') {
return value * 1000;
}
return String(value);
}
getColumnDefinitions(): ColumnDef[] {
return [
{
title: 'Process Name',
columnId: 'process_name',
},
{
title: 'PID',
columnId: 'pid',
formatHint: 'NUMERIC',
},
{
title: `Active power (estimated ${this.powerUnits()})`,
columnId: 'active_mw',
sum: true,
cellRenderer: this.renderMilliwatts.bind(this),
},
{
title: `Active energy (estimated ${this.powerUnits()}s)`,
columnId: 'active_mws',
sum: true,
cellRenderer: this.renderMilliwatts.bind(this),
sort: 'DESC',
},
{
title: `Idle transitions overhead (estimated ${this.powerUnits()}s)`,
columnId: 'idle_cost_mws',
sum: false,
cellRenderer: this.renderMilliwatts.bind(this),
},
{
title: `Total energy (estimated ${this.powerUnits()}s)`,
columnId: 'total_mws',
sum: true,
cellRenderer: this.renderMilliwatts.bind(this),
},
{
title: '% of total energy',
formatHint: 'PERCENT',
columnId: 'percent_of_total_energy',
sum: false,
},
];
}
getTabName() {
return 'Wattson by process';
}
}