blob: f2325cb19df19c8053ef5fa9d243d2498ee69cf2 [file] [log] [blame]
// 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 {exists} from '../../base/utils';
import {ColumnDef, Sorting} from '../../public/aggregation';
import {AreaSelection, AreaSelectionAggregator} from '../../public/selection';
import {CPU_SLICE_TRACK_KIND} from '../../public/track_kinds';
import {Engine} from '../../trace_processor/engine';
export class WattsonProcessSelectionAggregator
implements AreaSelectionAggregator
{
readonly id = 'wattson_process_aggregation';
async createAggregateView(engine: Engine, area: AreaSelection) {
await engine.query(`drop view if exists ${this.id};`);
const selectedCpus: number[] = [];
for (const trackInfo of area.tracks) {
trackInfo?.tags?.kind === CPU_SLICE_TRACK_KIND &&
exists(trackInfo.tags.cpu) &&
selectedCpus.push(trackInfo.tags.cpu);
}
if (selectedCpus.length === 0) return false;
const cpusCsv = `(` + selectedCpus.join() + `)`;
const duration = area.end - area.start;
// Prerequisite tables are already generated by Wattson thread aggregation,
// which is run prior to execution of this module
engine.query(`
-- Only get idle attribution in user defined window and filter by selected
-- CPUs and GROUP BY process
CREATE OR REPLACE PERFETTO TABLE _per_process_idle_attribution AS
SELECT
ROUND(SUM(idle_cost_mws), 2) as idle_cost_mws,
upid
FROM _filter_idle_attribution(${area.start}, ${duration})
WHERE cpu in ${cpusCsv}
GROUP BY upid;
-- Grouped by UPID and made CPU agnostic
CREATE VIEW ${this.id} AS
SELECT
ROUND(SUM(total_pws) / ${duration}, 2) as active_mw,
ROUND(SUM(total_pws) / 1000000000, 2) as active_mws,
COALESCE(idle_cost_mws, 0) as idle_cost_mws,
pid,
process_name
FROM _unioned_per_cpu_total
LEFT JOIN _per_process_idle_attribution USING (upid)
GROUP BY upid;
`);
return true;
}
getColumnDefinitions(): ColumnDef[] {
return [
{
title: 'Process Name',
kind: 'STRING',
columnConstructor: Uint16Array,
columnId: 'process_name',
},
{
title: 'PID',
kind: 'NUMBER',
columnConstructor: Uint16Array,
columnId: 'pid',
},
{
title: 'Active power (estimated mW)',
kind: 'NUMBER',
columnConstructor: Float64Array,
columnId: 'active_mw',
sum: true,
},
{
title: 'Active energy (estimated mWs)',
kind: 'NUMBER',
columnConstructor: Float64Array,
columnId: 'active_mws',
sum: true,
},
{
title: 'Idle transitions overhead (estimated mWs)',
kind: 'NUMBER',
columnConstructor: Float64Array,
columnId: 'idle_cost_mws',
sum: true,
},
];
}
async getExtra() {}
getTabName() {
return 'Wattson by process';
}
getDefaultSorting(): Sorting {
return {column: 'active_mws', direction: 'DESC'};
}
}