blob: e38d9b578cd961081e1f6b30745aeb58990d4e29 [file] [log] [blame]
// 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 m from 'mithril';
import {copyToClipboard} from '../../base/clipboard';
import {Icons} from '../../base/semantic_icons';
import {exists} from '../../base/utils';
import {addEphemeralTab} from '../../common/add_ephemeral_tab';
import {Upid} from '../../trace_processor/sql_utils/core_types';
import {
getProcessInfo,
getProcessName,
ProcessInfo,
} from '../../trace_processor/sql_utils/process';
import {Anchor} from '../../widgets/anchor';
import {MenuItem, PopupMenu2} from '../../widgets/menu';
import {ProcessDetailsTab} from '../process_details_tab';
import {
createSqlIdRefRenderer,
sqlIdRegistry,
} from './sql/details/sql_ref_renderer_registry';
import {asUpid} from '../../trace_processor/sql_utils/core_types';
import {AppImpl} from '../../core/app_trace_impl';
export function showProcessDetailsMenuItem(
upid: Upid,
pid?: number,
): m.Children {
return m(MenuItem, {
icon: Icons.ExternalLink,
label: 'Show process details',
onclick: () => {
// TODO(primiano): `trace` should be injected, but doing so would require
// an invasive refactoring of most classes in frontend/widgets/sql/*.
const trace = AppImpl.instance.trace;
if (trace === undefined) return;
addEphemeralTab(
'processDetails',
new ProcessDetailsTab({
trace,
upid,
pid,
}),
);
},
});
}
export function processRefMenuItems(info: {
upid: Upid;
name?: string;
pid?: number;
}): m.Children {
// We capture a copy to be able to pass it across async boundary to `onclick`.
const name = info.name;
return [
exists(name) &&
m(MenuItem, {
icon: Icons.Copy,
label: 'Copy process name',
onclick: () => copyToClipboard(name),
}),
exists(info.pid) &&
m(MenuItem, {
icon: Icons.Copy,
label: 'Copy pid',
onclick: () => copyToClipboard(`${info.pid}`),
}),
m(MenuItem, {
icon: Icons.Copy,
label: 'Copy upid',
onclick: () => copyToClipboard(`${info.upid}`),
}),
showProcessDetailsMenuItem(info.upid, info.pid),
];
}
export function renderProcessRef(info: ProcessInfo): m.Children {
return m(
PopupMenu2,
{
trigger: m(Anchor, getProcessName(info)),
},
processRefMenuItems(info),
);
}
sqlIdRegistry['process'] = createSqlIdRefRenderer<ProcessInfo>(
async (engine, id) => await getProcessInfo(engine, asUpid(Number(id))),
(data: ProcessInfo) => ({
value: renderProcessRef(data),
}),
);