blob: 0d55ff1aa9a37a091115d9d88a438394eec5f0d2 [file] [log] [blame]
// Copyright (C) 2023 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 {BigintMath as BIMath} from '../../base/bigint_math';
import {clamp} from '../../base/math_utils';
import {NAMED_ROW, NamedSliceTrack} from '../../frontend/named_slice_track';
import {SLICE_LAYOUT_FIT_CONTENT_DEFAULTS} from '../../frontend/slice_layout';
import {TrackEventDetails} from '../../public/selection';
import {Trace} from '../../public/trace';
import {Slice} from '../../public/track';
import {SourceDataset, Dataset} from '../../trace_processor/dataset';
import {
LONG,
LONG_NULL,
NUM,
NUM_NULL,
STR,
} from '../../trace_processor/query_result';
export const THREAD_SLICE_ROW = {
// Base columns (tsq, ts, dur, id, depth).
...NAMED_ROW,
// Thread-specific columns.
threadDur: LONG_NULL,
};
export type ThreadSliceRow = typeof THREAD_SLICE_ROW;
export class AsyncSliceTrack extends NamedSliceTrack<Slice, ThreadSliceRow> {
constructor(
trace: Trace,
uri: string,
maxDepth: number,
private readonly trackIds: number[],
) {
super(trace, uri);
this.sliceLayout = {
...SLICE_LAYOUT_FIT_CONTENT_DEFAULTS,
depthGuess: maxDepth,
};
}
getRowSpec(): ThreadSliceRow {
return THREAD_SLICE_ROW;
}
rowToSlice(row: ThreadSliceRow): Slice {
const namedSlice = this.rowToSliceBase(row);
if (row.dur > 0n && row.threadDur !== null) {
const fillRatio = clamp(BIMath.ratio(row.threadDur, row.dur), 0, 1);
return {...namedSlice, fillRatio};
} else {
return namedSlice;
}
}
getSqlSource(): string {
// If we only have one track ID we can avoid the overhead of
// experimental_slice_layout, and just go straight to the slice table.
if (this.trackIds.length === 1) {
return `
select
ts,
dur,
id,
depth,
ifnull(name, '[null]') as name,
thread_dur as threadDur
from slice
where track_id = ${this.trackIds[0]}
`;
} else {
return `
select
id,
ts,
dur,
layout_depth as depth,
ifnull(name, '[null]') as name,
thread_dur as threadDur
from experimental_slice_layout
where filter_track_ids = '${this.trackIds.join(',')}'
`;
}
}
onUpdatedSlices(slices: Slice[]) {
for (const slice of slices) {
slice.isHighlighted = slice === this.hoveredSlice;
}
}
async getSelectionDetails(
id: number,
): Promise<TrackEventDetails | undefined> {
const baseDetails = await super.getSelectionDetails(id);
if (!baseDetails) return undefined;
return {
...baseDetails,
tableName: 'slice',
};
}
override getDataset(): Dataset {
return new SourceDataset({
src: `slice`,
filter: {
col: 'track_id',
in: this.trackIds,
},
schema: {
id: NUM,
name: STR,
ts: LONG,
dur: LONG,
parent_id: NUM_NULL,
},
});
}
}