blob: 752684a89b238d988873c78fdc8c8fc6051dd859 [file] [log] [blame]
// Copyright (C) 2021 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 {searchSegment} from '../../base/binary_search';
import {duration, Time, time} from '../../base/time';
import {Actions} from '../../common/actions';
import {ProfileType, getLegacySelection} from '../../common/state';
import {TrackData} from '../../common/track_data';
import {TimelineFetcher} from '../../common/track_helper';
import {FLAMEGRAPH_HOVERED_COLOR} from '../../frontend/flamegraph';
import {FlamegraphDetailsPanel} from '../../frontend/flamegraph_panel';
import {globals} from '../../frontend/globals';
import {PanelSize} from '../../frontend/panel';
import {TimeScale} from '../../frontend/time_scale';
import {
EngineProxy,
Plugin,
PluginContextTrace,
PluginDescriptor,
Track,
} from '../../public';
import {LONG, NUM} from '../../trace_processor/query_result';
export const PERF_SAMPLES_PROFILE_TRACK_KIND = 'PerfSamplesProfileTrack';
export interface Data extends TrackData {
tsStarts: BigInt64Array;
}
const PERP_SAMPLE_COLOR = 'hsl(224, 45%, 70%)';
// 0.5 Makes the horizontal lines sharp.
const MARGIN_TOP = 4.5;
const RECT_HEIGHT = 30.5;
class PerfSamplesProfileTrack implements Track {
private centerY = this.getHeight() / 2;
private markerWidth = (this.getHeight() - MARGIN_TOP) / 2;
private hoveredTs: time | undefined = undefined;
private fetcher = new TimelineFetcher(this.onBoundsChange.bind(this));
private upid: number;
private engine: EngineProxy;
constructor(engine: EngineProxy, upid: number) {
this.upid = upid;
this.engine = engine;
}
async onUpdate(): Promise<void> {
await this.fetcher.requestDataForCurrentTime();
}
async onDestroy(): Promise<void> {
this.fetcher.dispose();
}
async onBoundsChange(
start: time,
end: time,
resolution: duration,
): Promise<Data> {
if (this.upid === undefined) {
return {
start,
end,
resolution,
length: 0,
tsStarts: new BigInt64Array(),
};
}
const queryRes = await this.engine.query(`
select ts, upid from perf_sample
join thread using (utid)
where upid = ${this.upid}
and callsite_id is not null
order by ts`);
const numRows = queryRes.numRows();
const data: Data = {
start,
end,
resolution,
length: numRows,
tsStarts: new BigInt64Array(numRows),
};
const it = queryRes.iter({ts: LONG});
for (let row = 0; it.valid(); it.next(), row++) {
data.tsStarts[row] = it.ts;
}
return data;
}
getHeight() {
return MARGIN_TOP + RECT_HEIGHT - 1;
}
render(ctx: CanvasRenderingContext2D, _size: PanelSize): void {
const {visibleTimeScale} = globals.timeline;
const data = this.fetcher.data;
if (data === undefined) return;
for (let i = 0; i < data.tsStarts.length; i++) {
const centerX = Time.fromRaw(data.tsStarts[i]);
const selection = getLegacySelection(globals.state);
const isHovered = this.hoveredTs === centerX;
const isSelected =
selection !== null &&
selection.kind === 'PERF_SAMPLES' &&
selection.leftTs <= centerX &&
selection.rightTs >= centerX;
const strokeWidth = isSelected ? 3 : 0;
this.drawMarker(
ctx,
visibleTimeScale.timeToPx(centerX),
this.centerY,
isHovered,
strokeWidth,
);
}
}
drawMarker(
ctx: CanvasRenderingContext2D,
x: number,
y: number,
isHovered: boolean,
strokeWidth: number,
): void {
ctx.beginPath();
ctx.moveTo(x, y - this.markerWidth);
ctx.lineTo(x - this.markerWidth, y);
ctx.lineTo(x, y + this.markerWidth);
ctx.lineTo(x + this.markerWidth, y);
ctx.lineTo(x, y - this.markerWidth);
ctx.closePath();
ctx.fillStyle = isHovered ? FLAMEGRAPH_HOVERED_COLOR : PERP_SAMPLE_COLOR;
ctx.fill();
if (strokeWidth > 0) {
ctx.strokeStyle = FLAMEGRAPH_HOVERED_COLOR;
ctx.lineWidth = strokeWidth;
ctx.stroke();
}
}
onMouseMove({x, y}: {x: number; y: number}) {
const data = this.fetcher.data;
if (data === undefined) return;
const {visibleTimeScale} = globals.timeline;
const time = visibleTimeScale.pxToHpTime(x);
const [left, right] = searchSegment(data.tsStarts, time.toTime());
const index = this.findTimestampIndex(
left,
visibleTimeScale,
data,
x,
y,
right,
);
this.hoveredTs =
index === -1 ? undefined : Time.fromRaw(data.tsStarts[index]);
}
onMouseOut() {
this.hoveredTs = undefined;
}
onMouseClick({x, y}: {x: number; y: number}) {
const data = this.fetcher.data;
if (data === undefined) return false;
const {visibleTimeScale} = globals.timeline;
const time = visibleTimeScale.pxToHpTime(x);
const [left, right] = searchSegment(data.tsStarts, time.toTime());
const index = this.findTimestampIndex(
left,
visibleTimeScale,
data,
x,
y,
right,
);
if (index !== -1) {
const ts = Time.fromRaw(data.tsStarts[index]);
globals.makeSelection(
Actions.selectPerfSamples({
id: index,
upid: this.upid,
leftTs: ts,
rightTs: ts,
type: ProfileType.PERF_SAMPLE,
}),
);
return true;
}
return false;
}
// If the markers overlap the rightmost one will be selected.
findTimestampIndex(
left: number,
timeScale: TimeScale,
data: Data,
x: number,
y: number,
right: number,
): number {
let index = -1;
if (left !== -1) {
const start = Time.fromRaw(data.tsStarts[left]);
const centerX = timeScale.timeToPx(start);
if (this.isInMarker(x, y, centerX)) {
index = left;
}
}
if (right !== -1) {
const start = Time.fromRaw(data.tsStarts[right]);
const centerX = timeScale.timeToPx(start);
if (this.isInMarker(x, y, centerX)) {
index = right;
}
}
return index;
}
isInMarker(x: number, y: number, centerX: number) {
return (
Math.abs(x - centerX) + Math.abs(y - this.centerY) <= this.markerWidth
);
}
}
class PerfSamplesProfilePlugin implements Plugin {
async onTraceLoad(ctx: PluginContextTrace): Promise<void> {
const result = await ctx.engine.query(`
select distinct upid, pid
from perf_sample join thread using (utid) join process using (upid)
where callsite_id is not null
`);
for (const it = result.iter({upid: NUM, pid: NUM}); it.valid(); it.next()) {
const upid = it.upid;
const pid = it.pid;
ctx.registerTrack({
uri: `perfetto.PerfSamplesProfile#${upid}`,
displayName: `Callstacks ${pid}`,
kind: PERF_SAMPLES_PROFILE_TRACK_KIND,
upid,
trackFactory: () => new PerfSamplesProfileTrack(ctx.engine, upid),
});
}
ctx.registerDetailsPanel({
render: (sel) => {
if (sel.kind === 'PERF_SAMPLES') {
return m(FlamegraphDetailsPanel);
} else {
return undefined;
}
},
});
}
}
export const plugin: PluginDescriptor = {
pluginId: 'perfetto.PerfSamplesProfile',
plugin: PerfSamplesProfilePlugin,
};