blob: 4af4015e02f35748b2992fb0fad31c9a9a04cbbe [file] [log] [blame]
// Copyright (C) 2020 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use size 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 m from 'mithril';
import {Icons} from '../base/semantic_icons';
import {Actions} from '../common/actions';
import {getLegacySelection} from '../common/state';
import {raf} from '../core/raf_scheduler';
import {Flow, globals} from './globals';
import {DurationWidget} from './widgets/duration';
import {EmptyState} from '../widgets/empty_state';
export const ALL_CATEGORIES = '_all_';
export function getFlowCategories(flow: Flow): string[] {
const categories: string[] = [];
// v1 flows have their own categories
if (flow.category) {
categories.push(...flow.category.split(','));
return categories;
}
const beginCats = flow.begin.sliceCategory.split(',');
const endCats = flow.end.sliceCategory.split(',');
categories.push(...new Set([...beginCats, ...endCats]));
return categories;
}
export class FlowEventsPanel implements m.ClassComponent {
view() {
const selection = getLegacySelection(globals.state);
if (!selection) {
return m(
EmptyState,
{
className: 'pf-noselection',
title: 'Nothing selected',
},
'Flow data will appear here',
);
}
if (selection.kind !== 'CHROME_SLICE') {
return m(
EmptyState,
{
className: 'pf-noselection',
title: 'No flow data',
icon: 'warning',
},
`Flows are not applicable to the selection kind: '${selection.kind}'`,
);
}
const flowClickHandler = (sliceId: number, trackId: number) => {
const trackKey = globals.trackManager.trackKeyByTrackId.get(trackId);
if (trackKey) {
globals.setLegacySelection(
{
kind: 'CHROME_SLICE',
id: sliceId,
trackKey,
table: 'slice',
},
{
clearSearch: true,
pendingScrollId: undefined,
switchToCurrentSelectionTab: false,
},
);
}
};
// Can happen only for flow events version 1
const haveCategories =
globals.connectedFlows.filter((flow) => flow.category).length > 0;
const columns = [
m('th', 'Direction'),
m('th', 'Duration'),
m('th', 'Connected Slice ID'),
m('th', 'Connected Slice Name'),
m('th', 'Thread Out'),
m('th', 'Thread In'),
m('th', 'Process Out'),
m('th', 'Process In'),
];
if (haveCategories) {
columns.push(m('th', 'Flow Category'));
columns.push(m('th', 'Flow Name'));
}
const rows = [m('tr', columns)];
// Fill the table with all the directly connected flow events
globals.connectedFlows.forEach((flow) => {
if (
selection.id !== flow.begin.sliceId &&
selection.id !== flow.end.sliceId
) {
return;
}
const outgoing = selection.id === flow.begin.sliceId;
const otherEnd = outgoing ? flow.end : flow.begin;
const args = {
onclick: () => flowClickHandler(otherEnd.sliceId, otherEnd.trackId),
onmousemove: () =>
globals.dispatch(
Actions.setHighlightedSliceId({sliceId: otherEnd.sliceId}),
),
onmouseleave: () =>
globals.dispatch(Actions.setHighlightedSliceId({sliceId: -1})),
};
const data = [
m('td.flow-link', args, outgoing ? 'Outgoing' : 'Incoming'),
m('td.flow-link', args, m(DurationWidget, {dur: flow.dur})),
m('td.flow-link', args, otherEnd.sliceId.toString()),
m('td.flow-link', args, otherEnd.sliceName),
m('td.flow-link', args, flow.begin.threadName),
m('td.flow-link', args, flow.end.threadName),
m('td.flow-link', args, flow.begin.processName),
m('td.flow-link', args, flow.end.processName),
];
if (haveCategories) {
data.push(m('td.flow-info', flow.category || '-'));
data.push(m('td.flow-info', flow.name || '-'));
}
rows.push(m('tr', data));
});
return m('.details-panel', [
m('.details-panel-heading', m('h2', `Flow events`)),
m('.flow-events-table', m('table', rows)),
]);
}
}
export class FlowEventsAreaSelectedPanel implements m.ClassComponent {
view() {
const selection = getLegacySelection(globals.state);
if (!selection || selection.kind !== 'AREA') {
return;
}
const columns = [
m('th', 'Flow Category'),
m('th', 'Number of flows'),
m(
'th',
'Show',
m(
'a.warning',
m('i.material-icons', 'warning'),
m(
'.tooltip',
'Showing a large number of flows may impact performance.',
),
),
),
];
const rows = [m('tr', columns)];
const categoryToFlowsNum = new Map<string, number>();
globals.selectedFlows.forEach((flow) => {
const categories = getFlowCategories(flow);
categories.forEach((cat) => {
if (!categoryToFlowsNum.has(cat)) {
categoryToFlowsNum.set(cat, 0);
}
categoryToFlowsNum.set(cat, categoryToFlowsNum.get(cat)! + 1);
});
});
const allWasChecked = globals.visibleFlowCategories.get(ALL_CATEGORIES);
rows.push(
m('tr.sum', [
m('td.sum-data', 'All'),
m('td.sum-data', globals.selectedFlows.length),
m(
'td.sum-data',
m(
'i.material-icons',
{
onclick: () => {
if (allWasChecked) {
globals.visibleFlowCategories.clear();
} else {
categoryToFlowsNum.forEach((_, cat) => {
globals.visibleFlowCategories.set(cat, true);
});
}
globals.visibleFlowCategories.set(
ALL_CATEGORIES,
!allWasChecked,
);
raf.scheduleFullRedraw();
},
},
allWasChecked ? Icons.Checkbox : Icons.BlankCheckbox,
),
),
]),
);
categoryToFlowsNum.forEach((num, cat) => {
const wasChecked =
globals.visibleFlowCategories.get(cat) ||
globals.visibleFlowCategories.get(ALL_CATEGORIES);
const data = [
m('td.flow-info', cat),
m('td.flow-info', num),
m(
'td.flow-info',
m(
'i.material-icons',
{
onclick: () => {
if (wasChecked) {
globals.visibleFlowCategories.set(ALL_CATEGORIES, false);
}
globals.visibleFlowCategories.set(cat, !wasChecked);
raf.scheduleFullRedraw();
},
},
wasChecked ? Icons.Checkbox : Icons.BlankCheckbox,
),
),
];
rows.push(m('tr', data));
});
return m('.details-panel', [
m('.details-panel-heading', m('h2', `Selected flow events`)),
m('.flow-events-table', m('table', rows)),
]);
}
}