blob: b6efe68e87bf0bce4526d56d380020d8372c06b9 [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 {addWattsonThreadTrack} from './wattson_thread_utils';
import type {
Aggregation,
Aggregator,
} from '../../components/aggregation_adapter';
import type {AreaSelection} from '../../public/selection';
import {Button, ButtonVariant} from '../../widgets/button';
import {CPU_SLICE_TRACK_KIND} from '../../public/track_kinds';
import type {Engine} from '../../trace_processor/engine';
import {Intent} from '../../widgets/common';
import type {SqlValue} from '../../trace_processor/query_result';
import {RadioGroup} from '../../widgets/radio_group';
import type {Trace} from '../../public/trace';
import {WATTSON_THREAD_TRACK_KIND} from './track_kinds';
export class WattsonThreadSelectionAggregator implements Aggregator {
readonly id = 'wattson_plugin_thread_aggregation';
private scaleNumericData: boolean = false;
constructor(private trace: Trace) {}
probe(area: AreaSelection): Aggregation | undefined {
const selectedCpus: number[] = [];
const selectedUtids: number[] = [];
for (const trackInfo of area.tracks) {
if (trackInfo?.tags?.kinds?.includes(CPU_SLICE_TRACK_KIND)) {
exists(trackInfo.tags.cpu) && selectedCpus.push(trackInfo.tags.cpu);
}
if (trackInfo?.tags?.kinds?.includes(WATTSON_THREAD_TRACK_KIND)) {
exists(trackInfo.tags.utid) && selectedUtids.push(trackInfo.tags.utid);
}
}
if (selectedCpus.length === 0 && selectedUtids.length === 0) {
return undefined;
}
return {
prepareData: async (engine: Engine) => {
await engine.query(`drop view if exists ${this.id};`);
const duration = area.end - area.start;
const filters = [];
if (selectedCpus.length > 0) {
filters.push(`cpu IN (${selectedCpus.join()})`);
}
if (selectedUtids.length > 0) {
filters.push(`utid IN (${selectedUtids.join()})`);
}
const whereClause = `WHERE ${filters.join(' OR ')}`;
await engine.query(`
INCLUDE PERFETTO MODULE wattson.aggregation;
CREATE OR REPLACE PERFETTO TABLE wattson_plugin_ui_selection_window AS
SELECT
${area.start} as ts,
${duration} as dur,
0 as period_id;
-- Prefilter tasks table to be small
CREATE OR REPLACE PERFETTO VIEW _wattson_ui_selected_tasks AS
SELECT *
FROM _estimates_w_tasks_attribution
${whereClause};
-- Use a dedicated CPUs table to avoid incorrectly filtering idle costs
CREATE OR REPLACE PERFETTO TABLE _wattson_ui_selected_cpus AS
SELECT cpu FROM _wattson_cpus
${selectedCpus.length > 0 ? `WHERE cpu IN (${selectedCpus.join()})` : ''};
-- Use SPAN_JOIN to clip tasks to the window
DROP TABLE IF EXISTS _wattson_ui_windowed_tasks;
CREATE VIRTUAL TABLE _wattson_ui_windowed_tasks
USING SPAN_JOIN(
wattson_plugin_ui_selection_window,
_wattson_ui_selected_tasks
);
-- Materialize the thread-level summary once.
CREATE OR REPLACE PERFETTO TABLE wattson_plugin_thread_summary AS
SELECT *
FROM _wattson_threads_aggregation!(
_wattson_ui_windowed_tasks,
wattson_plugin_ui_selection_window,
_wattson_ui_selected_cpus
);
CREATE PERFETTO VIEW ${this.id} AS
WITH base AS (
SELECT
ROUND(estimated_mw, 3) as active_mw,
ROUND(estimated_mws, 3) as active_mws,
ROUND(idle_transitions_mws, 3) as idle_cost_mws,
ROUND(total_mws, 3) as total_mws,
thread_name,
utid,
tid,
pid
FROM wattson_plugin_thread_summary
)
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);
}
private renderShowButton(utid: SqlValue): m.Children {
return m(Button, {
label: 'Show',
intent: Intent.Primary,
variant: ButtonVariant.Filled,
compact: true,
onclick: () => {
const utidNum = typeof utid === 'number' ? utid : Number(utid);
addWattsonThreadTrack(this.trace, utidNum);
},
});
}
getColumnDefinitions(): ColumnDef[] {
return [
{
title: 'Track',
columnId: 'utid',
cellRenderer: this.renderShowButton.bind(this),
},
{
title: 'Thread Name',
columnId: 'thread_name',
},
{
title: 'TID',
columnId: 'tid',
formatHint: 'NUMERIC',
},
{
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 thread';
}
}