| // 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 { |
| createQueryCounterTrack, |
| SqlDataSource, |
| } from '../../components/tracks/query_counter_track'; |
| import {PerfettoPlugin} from '../../public/plugin'; |
| import {Trace} from '../../public/trace'; |
| import {COUNTER_TRACK_KIND, SLICE_TRACK_KIND} from '../../public/track_kinds'; |
| import {TrackNode} from '../../public/workspace'; |
| import {NUM, NUM_NULL, STR} from '../../trace_processor/query_result'; |
| import ProcessThreadGroupsPlugin from '../dev.perfetto.ProcessThreadGroups'; |
| import StandardGroupsPlugin from '../dev.perfetto.StandardGroups'; |
| import TraceProcessorTrackPlugin from '../dev.perfetto.TraceProcessorTrack'; |
| import {TraceProcessorCounterTrack} from '../dev.perfetto.TraceProcessorTrack/trace_processor_counter_track'; |
| import {createTraceProcessorSliceTrack} from '../dev.perfetto.TraceProcessorTrack/trace_processor_slice_track'; |
| |
| async function registerAllocsTrack( |
| ctx: Trace, |
| uri: string, |
| dataSource: SqlDataSource, |
| ) { |
| const track = await createQueryCounterTrack({ |
| trace: ctx, |
| uri, |
| data: dataSource, |
| }); |
| ctx.tracks.registerTrack({ |
| uri, |
| title: `dmabuf allocs`, |
| track: track, |
| }); |
| } |
| |
| export default class implements PerfettoPlugin { |
| static readonly id = 'dev.perfetto.AndroidDmabuf'; |
| static readonly dependencies = [ |
| ProcessThreadGroupsPlugin, |
| StandardGroupsPlugin, |
| TraceProcessorTrackPlugin, |
| ]; |
| |
| async onTraceLoad(ctx: Trace): Promise<void> { |
| const e = ctx.engine; |
| await e.query(`INCLUDE PERFETTO MODULE android.memory.dmabuf`); |
| await e.query(` |
| CREATE PERFETTO TABLE _android_memory_cumulative_dmabuf AS |
| SELECT |
| upid, utid, ts, |
| SUM(buf_size) OVER(PARTITION BY COALESCE(upid, utid) ORDER BY ts) AS value |
| FROM android_dmabuf_allocs;`); |
| |
| const pids = await e.query( |
| `SELECT DISTINCT upid, IIF(upid IS NULL, utid, NULL) AS utid FROM _android_memory_cumulative_dmabuf`, |
| ); |
| const it = pids.iter({upid: NUM_NULL, utid: NUM_NULL}); |
| for (; it.valid(); it.next()) { |
| if (it.upid != null) { |
| const uri = `/android_process_dmabuf_upid_${it.upid}`; |
| const config: SqlDataSource = { |
| sqlSource: `SELECT ts, value FROM _android_memory_cumulative_dmabuf |
| WHERE upid = ${it.upid}`, |
| }; |
| await registerAllocsTrack(ctx, uri, config); |
| ctx.plugins |
| .getPlugin(ProcessThreadGroupsPlugin) |
| .getGroupForProcess(it.upid) |
| ?.addChildInOrder(new TrackNode({uri, title: 'dmabuf allocs'})); |
| } else if (it.utid != null) { |
| const uri = `/android_process_dmabuf_utid_${it.utid}`; |
| const config: SqlDataSource = { |
| sqlSource: `SELECT ts, value FROM _android_memory_cumulative_dmabuf |
| WHERE utid = ${it.utid}`, |
| }; |
| await registerAllocsTrack(ctx, uri, config); |
| ctx.plugins |
| .getPlugin(ProcessThreadGroupsPlugin) |
| .getGroupForThread(it.utid) |
| ?.addChildInOrder(new TrackNode({uri, title: 'dmabuf allocs'})); |
| } |
| } |
| const memoryGroupFn = () => { |
| return ctx.plugins |
| .getPlugin(StandardGroupsPlugin) |
| .getOrCreateStandardGroup(ctx.workspace, 'MEMORY'); |
| }; |
| const node = await addGlobalCounter(ctx, memoryGroupFn); |
| await addGlobalAllocs(ctx, () => { |
| return node ?? memoryGroupFn(); |
| }); |
| } |
| } |
| |
| async function addGlobalCounter(ctx: Trace, parent: () => TrackNode) { |
| const track = await ctx.engine.query(` |
| select id, name |
| from track |
| where type = 'android_dma_heap' |
| `); |
| const it = track.maybeFirstRow({id: NUM, name: STR}); |
| if (!it) { |
| return undefined; |
| } |
| const {id, name: title} = it; |
| const uri = `/android_dmabuf_counter`; |
| ctx.tracks.registerTrack({ |
| uri, |
| title, |
| tags: { |
| kind: COUNTER_TRACK_KIND, |
| trackIds: [id], |
| }, |
| track: new TraceProcessorCounterTrack(ctx, uri, {}, id, title), |
| }); |
| const node = new TrackNode({ |
| uri, |
| title, |
| }); |
| parent().addChildInOrder(node); |
| return node; |
| } |
| |
| async function addGlobalAllocs(ctx: Trace, parent: () => TrackNode) { |
| const track = await ctx.engine.query(` |
| select name, group_concat(id) as trackIds |
| from track |
| where type = 'android_dma_allocations' |
| group by name |
| `); |
| const it = track.maybeFirstRow({trackIds: STR, name: STR}); |
| if (!it) { |
| return undefined; |
| } |
| const {trackIds, name: title} = it; |
| const uri = `/android_dmabuf_allocs`; |
| const ids = trackIds.split(',').map((x) => Number(x)); |
| ctx.tracks.registerTrack({ |
| uri, |
| title, |
| tags: { |
| kind: SLICE_TRACK_KIND, |
| trackIds: ids, |
| }, |
| track: createTraceProcessorSliceTrack({trace: ctx, uri, trackIds: ids}), |
| }); |
| const node = new TrackNode({ |
| uri, |
| title, |
| }); |
| parent().addChildInOrder(node); |
| } |