1*6dbdd20aSAndroid Build Coastguard Worker// Copyright (C) 2024 The Android Open Source Project 2*6dbdd20aSAndroid Build Coastguard Worker// 3*6dbdd20aSAndroid Build Coastguard Worker// Licensed under the Apache License, Version 2.0 (the "License"); 4*6dbdd20aSAndroid Build Coastguard Worker// you may not use this file except in compliance with the License. 5*6dbdd20aSAndroid Build Coastguard Worker// You may obtain a copy of the License at 6*6dbdd20aSAndroid Build Coastguard Worker// 7*6dbdd20aSAndroid Build Coastguard Worker// http://www.apache.org/licenses/LICENSE-2.0 8*6dbdd20aSAndroid Build Coastguard Worker// 9*6dbdd20aSAndroid Build Coastguard Worker// Unless required by applicable law or agreed to in writing, software 10*6dbdd20aSAndroid Build Coastguard Worker// distributed under the License is distributed on an "AS IS" BASIS, 11*6dbdd20aSAndroid Build Coastguard Worker// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12*6dbdd20aSAndroid Build Coastguard Worker// See the License for the specific language governing permissions and 13*6dbdd20aSAndroid Build Coastguard Worker// limitations under the License. 14*6dbdd20aSAndroid Build Coastguard Worker 15*6dbdd20aSAndroid Build Coastguard Workerimport {TrackNode} from '../../public/workspace'; 16*6dbdd20aSAndroid Build Coastguard Workerimport {Trace} from '../../public/trace'; 17*6dbdd20aSAndroid Build Coastguard Workerimport {PerfettoPlugin} from '../../public/plugin'; 18*6dbdd20aSAndroid Build Coastguard Workerimport {createQueryCounterTrack} from '../../components/tracks/query_counter_track'; 19*6dbdd20aSAndroid Build Coastguard Workerimport {Engine} from '../../trace_processor/engine'; 20*6dbdd20aSAndroid Build Coastguard Workerimport {NUM} from '../../trace_processor/query_result'; 21*6dbdd20aSAndroid Build Coastguard Worker 22*6dbdd20aSAndroid Build Coastguard Workerexport default class implements PerfettoPlugin { 23*6dbdd20aSAndroid Build Coastguard Worker static readonly id = 'dev.perfetto.Io'; 24*6dbdd20aSAndroid Build Coastguard Worker async onTraceLoad(ctx: Trace) { 25*6dbdd20aSAndroid Build Coastguard Worker await ctx.engine.query(`INCLUDE PERFETTO MODULE linux.block_io`); 26*6dbdd20aSAndroid Build Coastguard Worker const devices = await this.lookupDevices(ctx.engine); 27*6dbdd20aSAndroid Build Coastguard Worker const group = new TrackNode({ 28*6dbdd20aSAndroid Build Coastguard Worker title: 'Queued IO requests', 29*6dbdd20aSAndroid Build Coastguard Worker sortOrder: -5, 30*6dbdd20aSAndroid Build Coastguard Worker isSummary: true, 31*6dbdd20aSAndroid Build Coastguard Worker }); 32*6dbdd20aSAndroid Build Coastguard Worker for (const device of devices) { 33*6dbdd20aSAndroid Build Coastguard Worker const uri = `/queued_io_request_count/device_${device['id']}`; 34*6dbdd20aSAndroid Build Coastguard Worker const title = `dev major:${device['major']} minor:${device['minor']}`; 35*6dbdd20aSAndroid Build Coastguard Worker const track = await createQueryCounterTrack({ 36*6dbdd20aSAndroid Build Coastguard Worker trace: ctx, 37*6dbdd20aSAndroid Build Coastguard Worker uri, 38*6dbdd20aSAndroid Build Coastguard Worker data: { 39*6dbdd20aSAndroid Build Coastguard Worker sqlSource: 40*6dbdd20aSAndroid Build Coastguard Worker `SELECT ts, ops_in_queue_or_device as value 41*6dbdd20aSAndroid Build Coastguard Worker FROM linux_active_block_io_operations_by_device 42*6dbdd20aSAndroid Build Coastguard Worker WHERE dev = ${String(device['id'])}` 43*6dbdd20aSAndroid Build Coastguard Worker }, 44*6dbdd20aSAndroid Build Coastguard Worker }); 45*6dbdd20aSAndroid Build Coastguard Worker ctx.tracks.registerTrack({ 46*6dbdd20aSAndroid Build Coastguard Worker uri, 47*6dbdd20aSAndroid Build Coastguard Worker title, 48*6dbdd20aSAndroid Build Coastguard Worker tags: { 49*6dbdd20aSAndroid Build Coastguard Worker device: device['id'], 50*6dbdd20aSAndroid Build Coastguard Worker groupName: 'Queued IO requests', 51*6dbdd20aSAndroid Build Coastguard Worker }, 52*6dbdd20aSAndroid Build Coastguard Worker track, 53*6dbdd20aSAndroid Build Coastguard Worker }); 54*6dbdd20aSAndroid Build Coastguard Worker const node = new TrackNode({uri, title}); 55*6dbdd20aSAndroid Build Coastguard Worker group.addChildInOrder(node); 56*6dbdd20aSAndroid Build Coastguard Worker } 57*6dbdd20aSAndroid Build Coastguard Worker if (group.children.length) { 58*6dbdd20aSAndroid Build Coastguard Worker ctx.workspace.addChildInOrder(group); 59*6dbdd20aSAndroid Build Coastguard Worker } 60*6dbdd20aSAndroid Build Coastguard Worker } 61*6dbdd20aSAndroid Build Coastguard Worker 62*6dbdd20aSAndroid Build Coastguard Worker private async lookupDevices(engine: Engine): Promise<{ [key: string]: number; }[]> { 63*6dbdd20aSAndroid Build Coastguard Worker const query =` 64*6dbdd20aSAndroid Build Coastguard Worker SELECT DISTINCT dev, linux_device_major_id(dev) as major, linux_device_minor_id(dev) as minor 65*6dbdd20aSAndroid Build Coastguard Worker FROM linux_active_block_io_operations_by_device ORDER BY dev`; 66*6dbdd20aSAndroid Build Coastguard Worker const result = await engine.query(query); 67*6dbdd20aSAndroid Build Coastguard Worker const it = result.iter({dev: NUM, major: NUM, minor: NUM}); 68*6dbdd20aSAndroid Build Coastguard Worker 69*6dbdd20aSAndroid Build Coastguard Worker const devs: { [key: string]: number; }[] = []; 70*6dbdd20aSAndroid Build Coastguard Worker 71*6dbdd20aSAndroid Build Coastguard Worker for (; it.valid(); it.next()) { 72*6dbdd20aSAndroid Build Coastguard Worker devs.push({'id': it.dev, 'major': it.major, 'minor': it.minor}); 73*6dbdd20aSAndroid Build Coastguard Worker } 74*6dbdd20aSAndroid Build Coastguard Worker 75*6dbdd20aSAndroid Build Coastguard Worker return devs; 76*6dbdd20aSAndroid Build Coastguard Worker } 77*6dbdd20aSAndroid Build Coastguard Worker} 78