blob: b3f8cf8b582de8cdfb79307ac50b43b9ea310eb1 [file]
// Copyright (C) 2018 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 {assertExists, assertTrue} from '../base/assert';
import {type time, Time, TimeSpan} from '../base/time';
import {cacheTrace} from './cache_manager';
import {
getEnabledMetatracingCategories,
isMetatracingEnabled,
} from './metatracing';
import {featureFlags} from './feature_flags';
import type {Engine, EngineBase} from '../trace_processor/engine';
import {HttpRpcEngine} from '../trace_processor/http_rpc_engine';
import {
LONG,
LONG_NULL,
NUM,
NUM_NULL,
STR,
} from '../trace_processor/query_result';
import {WasmEngineProxy} from '../trace_processor/wasm_engine_proxy';
import {
TraceBufferStream,
TraceFileStream,
TraceHttpStream,
TraceMultipleFilesStream,
} from '../core/trace_stream';
import type {TraceStream} from '../public/stream';
import {
deserializeAppStatePhase1,
deserializeAppStatePhase2,
} from './state_serialization';
import type {AppImpl} from './app_impl';
import {raf} from './raf_scheduler';
import {TraceImpl} from './trace_impl';
import type {TraceSource} from './trace_source';
import {Router} from '../core/router';
import type {TraceInfoImpl} from './trace_info_impl';
import {base64Decode} from '../base/string_utils';
import {
parseUrlCommands,
StartupCommandNotAllowedError,
} from './command_manager';
import {HighPrecisionTimeSpan} from '../base/high_precision_time_span';
import {sha1} from '../base/hash';
import {showModal} from '../widgets/modal';
import m from 'mithril';
const ENABLE_CHROME_RELIABLE_RANGE_ZOOM_FLAG = featureFlags.register({
id: 'enableChromeReliableRangeZoom',
name: 'Enable Chrome reliable range zoom',
description: 'Automatically zoom into the reliable range for Chrome traces',
defaultValue: false,
});
const ENABLE_CHROME_RELIABLE_RANGE_ANNOTATION_FLAG = featureFlags.register({
id: 'enableChromeReliableRangeAnnotation',
name: 'Enable Chrome reliable range annotation',
description: 'Automatically adds an annotation for the reliable range start',
defaultValue: false,
});
// The following flags control TraceProcessor Config.
const CROP_TRACK_EVENTS_FLAG = featureFlags.register({
id: 'cropTrackEvents',
name: 'Crop track events',
description: 'Ignores track events outside of the range of interest',
defaultValue: false,
});
const INGEST_FTRACE_IN_RAW_TABLE_FLAG = featureFlags.register({
id: 'ingestFtraceInRawTable',
name: 'Ingest ftrace in raw table',
description: 'Enables ingestion of typed ftrace events into the raw table',
defaultValue: true,
});
const ANALYZE_TRACE_PROTO_CONTENT_FLAG = featureFlags.register({
id: 'analyzeTraceProtoContent',
name: 'Analyze trace proto content',
description:
'Enables trace proto content analysis (experimental_proto_content table)',
defaultValue: false,
});
const FTRACE_DROP_UNTIL_FLAG = featureFlags.register({
id: 'ftraceDropUntilAllCpusValid',
name: 'Crop ftrace events',
description:
'Drop ftrace events until all per-cpu data streams are known to be valid',
defaultValue: true,
});
const FORCE_FULL_SORT_FLAG = featureFlags.register({
id: 'forceFullSort',
name: 'Force full sort',
description:
'Forces the trace processor into performing a full sort ignoring any windowing logic',
defaultValue: false,
});
// TODO(stevegolton): Move this into some global "SQL extensions" file and
// ensure it's only run once.
async function defineMaxLayoutDepthSqlFunction(engine: Engine): Promise<void> {
await engine.query(`
create perfetto function __max_layout_depth(track_count INT, track_ids STRING)
returns INT AS
select iif(
$track_count = 1,
(
select max_depth
from _slice_track_summary
where id = cast($track_ids AS int)
),
(
select max(layout_depth)
from experimental_slice_layout($track_ids)
)
);
`);
}
let lastEngineId = 0;
export async function loadTrace(
app: AppImpl,
traceSource: TraceSource,
): Promise<TraceImpl> {
updateStatus(app, 'Opening trace');
const engineId = `${++lastEngineId}`;
const engine = await createEngine(app, engineId);
return await loadTraceIntoEngine(app, traceSource, engine);
}
async function createEngine(
app: AppImpl,
engineId: string,
): Promise<EngineBase> {
// Check if there is any instance of the trace_processor_shell running in
// HTTP RPC mode (i.e. trace_processor_shell -D).
let useRpc = false;
if (app.httpRpc.newEngineMode === 'USE_HTTP_RPC_IF_AVAILABLE') {
useRpc = (await HttpRpcEngine.checkConnection()).connected;
}
const descriptorBlobs: Uint8Array[] = [];
for (const b64Str of await app.protoDescriptors()) {
descriptorBlobs.push(base64Decode(b64Str));
}
let engine;
if (useRpc) {
console.log('Opening trace using native accelerator over HTTP+RPC');
engine = new HttpRpcEngine(engineId);
} else {
console.log('Opening trace using built-in WASM engine');
engine = new WasmEngineProxy(engineId);
engine.resetTraceProcessor({
tokenizeOnly: false,
cropTrackEvents: CROP_TRACK_EVENTS_FLAG.get(),
ingestFtraceInRawTable: INGEST_FTRACE_IN_RAW_TABLE_FLAG.get(),
analyzeTraceProtoContent: ANALYZE_TRACE_PROTO_CONTENT_FLAG.get(),
ftraceDropUntilAllCpusValid: FTRACE_DROP_UNTIL_FLAG.get(),
extraParsingDescriptors: descriptorBlobs,
forceFullSort: FORCE_FULL_SORT_FLAG.get(),
});
}
engine.onResponseReceived = () => raf.scheduleFullRedraw();
if (isMetatracingEnabled()) {
engine.enableMetatrace(assertExists(getEnabledMetatracingCategories()));
}
return engine;
}
async function loadTraceIntoEngine(
app: AppImpl,
traceSource: TraceSource,
engine: EngineBase,
): Promise<TraceImpl> {
let traceStream: TraceStream | undefined;
const serializedAppState = traceSource.serializedAppState;
if (traceSource.type === 'FILE') {
traceStream = new TraceFileStream(traceSource.file);
} else if (traceSource.type === 'ARRAY_BUFFER') {
traceStream = new TraceBufferStream(traceSource.buffer);
} else if (traceSource.type === 'URL') {
traceStream = new TraceHttpStream(traceSource.url);
} else if (traceSource.type === 'STREAM') {
traceStream = traceSource.stream;
} else if (traceSource.type === 'HTTP_RPC') {
traceStream = undefined;
} else if (traceSource.type === 'MULTIPLE_FILES') {
traceStream = new TraceMultipleFilesStream(traceSource.files);
} else {
throw new Error(`Unknown source: ${JSON.stringify(traceSource)}`);
}
// |traceStream| can be undefined in the case when we are using the external
// HTTP+RPC endpoint and the trace processor instance has already loaded
// a trace (because it was passed as a cmdline argument to
// trace_processor_shell). In this case we don't want the UI to load any
// file/stream and we just want to jump to the loading phase.
if (traceStream !== undefined) {
const tStart = performance.now();
for (;;) {
const res = await traceStream.readChunk();
await engine.parse(res.data);
const elapsed = (performance.now() - tStart) / 1000;
let status = 'Loading trace ';
if (res.bytesTotal > 0) {
const progress = Math.round((res.bytesRead / res.bytesTotal) * 100);
status += `${progress}%`;
} else {
status += `${Math.round(res.bytesRead / 1e6)} MB`;
}
status += ` - ${Math.ceil(res.bytesRead / elapsed / 1e6)} MB/s`;
updateStatus(app, status);
if (res.eof) break;
}
await engine.notifyEof();
} else {
assertTrue(engine instanceof HttpRpcEngine);
await engine.restoreInitialTables();
}
for (const p of await app.sqlPackages()) {
await engine.registerSqlPackages(p);
}
const traceDetails = await getTraceInfo(engine, app, traceSource);
const trace = new TraceImpl(app, engine, traceDetails);
app.setActiveTrace(trace);
const hasJsonTrace = traceDetails.traceTypes.includes('json');
const visibleTimeSpan = await computeVisibleTime(
traceDetails.start,
traceDetails.end,
hasJsonTrace,
engine,
);
const newViewport = HighPrecisionTimeSpan.fromTime(
visibleTimeSpan.start,
visibleTimeSpan.end,
);
trace.timeline.setVisibleWindow(newViewport);
const cacheUuid = traceDetails.cached ? traceDetails.uuid : '';
// Make sure the helper views are available before we start adding tracks.
await includeSummaryTables(trace);
await defineMaxLayoutDepthSqlFunction(engine);
if (serializedAppState !== undefined) {
deserializeAppStatePhase1(serializedAppState, trace);
}
await app.plugins.onTraceLoad(trace, (id) => {
updateStatus(app, `Running plugin: ${id}`);
});
// Plugins may call trace.initialPage.suggest(...) during onTraceLoad to
// request that the app navigate somewhere other than /viewer.
const initialRoute = trace.initialPage.getWinner() ?? '/viewer';
Router.navigate(`#!${initialRoute}?local_cache_key=${cacheUuid}`);
decideTabs(trace);
updateStatus(app, `Loading minimap`);
await trace.minimap.load(traceDetails.start, traceDetails.end);
// Trace Processor doesn't support the reliable range feature for JSON
// traces.
if (!hasJsonTrace && ENABLE_CHROME_RELIABLE_RANGE_ANNOTATION_FLAG.get()) {
const reliableRangeStart = await computeTraceReliableRangeStart(engine);
if (reliableRangeStart > 0) {
trace.notes.addNote({
timestamp: reliableRangeStart,
color: '#ff0000',
text: 'Reliable Range Start',
});
}
}
// notify() will await that all listeners' promises have resolved.
await trace.onTraceReady.notify();
if (serializedAppState !== undefined) {
// Wait that plugins have completed their actions and then proceed with
// the final phase of app state restore.
// TODO(primiano): this can probably be removed once we refactor tracks
// to be URI based and can deal with non-existing URIs.
deserializeAppStatePhase2(serializedAppState, trace);
}
// Execute startup commands as the final step - simulates user actions
// after the trace is fully loaded and any saved state has been restored.
// This ensures startup commands see the complete, final state of the trace.
// CRITICAL ORDER: URL commands MUST execute before settings commands!
// This ordering has subtle but important implications:
// - URL commands are trace-specific and should establish initial state
// - Settings commands are user preferences that should override URL defaults
// - Changing this order could break trace sharing and user customization
// DO NOT REORDER without understanding the full impact!
const urlCommands =
parseUrlCommands(app.initialRouteArgs.startupCommands) ?? [];
const settingsCommands = app.startupCommandsSetting.get();
// Combine URL and settings commands - runtime allowlist checking will handle filtering
const allStartupCommands = [...urlCommands, ...settingsCommands];
const enforceAllowlist = app.enforceStartupCommandAllowlistSetting.get();
if (allStartupCommands.length > 0) {
updateStatus(app, 'Running startup commands');
using _ = trace.omnibox.disablePrompts();
// Execute startup commands in trace context after everything is ready.
// This simulates user actions taken after trace load is complete,
// including any saved app state restoration. At this point:
// - All plugins have loaded and registered their commands
// - Trace data is fully accessible
// - UI state has been restored from any saved workspace
// - Commands can safely query trace data and modify UI state
// Set allowlist checking during startup if enforcement enabled
if (enforceAllowlist) {
app.commands.setExecutingStartupCommands(true);
}
const blocked: string[] = [];
const failed: Array<{id: string; error: unknown}> = [];
try {
for (const command of allStartupCommands) {
try {
// Execute through proxy to access both global and trace-specific
// commands.
await app.commands.runCommand(command.id, ...command.args);
} catch (error) {
if (error instanceof StartupCommandNotAllowedError) {
blocked.push(error.commandId);
} else {
failed.push({id: command.id, error});
}
}
}
} finally {
// Always restore default (allow all) behavior when done
app.commands.setExecutingStartupCommands(false);
}
if (blocked.length > 0 || failed.length > 0) {
showStartupCommandIssuesDialog(blocked, failed);
}
}
return trace;
}
function showStartupCommandIssuesDialog(
blocked: ReadonlyArray<string>,
failed: ReadonlyArray<{id: string; error: unknown}>,
) {
const uniqueBlocked = Array.from(new Set(blocked));
showModal({
title: 'Some startup commands did not run',
content: () =>
m(
'.pf-startup-command-issues',
uniqueBlocked.length > 0 &&
m(
'section',
m(
'p',
'These commands were blocked because they are not on the ',
"allowlist. Disable 'Enforce startup command allowlist' in ",
'settings to run them anyway.',
),
m(
'ul',
uniqueBlocked.map((id) => m('li', m('code', id))),
),
),
failed.length > 0 &&
m(
'section',
m('p', 'These commands threw an error while executing:'),
m(
'ul',
failed.map(({id, error}) =>
m('li', m('code', id), ': ', String(error)),
),
),
),
),
buttons: [{text: 'Dismiss', primary: true}],
});
}
function decideTabs(trace: TraceImpl) {
// Show the list of default tabs, but don't make them active!
for (const tabUri of trace.tabs.defaultTabs) {
trace.tabs.showTab(tabUri);
}
}
async function includeSummaryTables(trace: TraceImpl) {
const engine = trace.engine;
updateStatus(trace, 'Creating slice summaries');
await engine.query(`include perfetto module viz.summary.slices;`);
updateStatus(trace, 'Creating counter summaries');
await engine.query(`include perfetto module viz.summary.counters;`);
updateStatus(trace, 'Creating thread summaries');
await engine.query(`include perfetto module viz.summary.threads;`);
updateStatus(trace, 'Creating processes summaries');
await engine.query(`include perfetto module viz.summary.processes;`);
}
function updateStatus(traceOrApp: TraceImpl | AppImpl, msg: string): void {
const showUntilDismissed = 0;
traceOrApp.omnibox.showStatusMessage(msg, showUntilDismissed);
}
async function computeFtraceBounds(engine: Engine): Promise<TimeSpan | null> {
const result = await engine.query(`
SELECT min(ts) as start, max(ts) as end FROM ftrace_event;
`);
const {start, end} = result.firstRow({start: LONG_NULL, end: LONG_NULL});
if (start !== null && end !== null) {
return new TimeSpan(Time.fromRaw(start), Time.fromRaw(end));
}
return null;
}
async function computeTraceReliableRangeStart(engine: Engine): Promise<time> {
const result =
await engine.query(`SELECT RUN_METRIC('chrome/chrome_reliable_range.sql');
SELECT start FROM chrome_reliable_range`);
const bounds = result.firstRow({start: LONG});
return Time.fromRaw(bounds.start);
}
async function computeVisibleTime(
traceStart: time,
traceEnd: time,
isJsonTrace: boolean,
engine: Engine,
): Promise<TimeSpan> {
// initialise visible time to the trace time bounds
let visibleStart = traceStart;
let visibleEnd = traceEnd;
// compare start and end with metadata computed by the trace processor
const mdTime = await getTracingMetadataTimeBounds(engine);
// make sure the bounds hold
if (Time.max(visibleStart, mdTime.start) < Time.min(visibleEnd, mdTime.end)) {
visibleStart = Time.max(visibleStart, mdTime.start);
visibleEnd = Time.min(visibleEnd, mdTime.end);
}
// Trace Processor doesn't support the reliable range feature for JSON
// traces.
if (!isJsonTrace && ENABLE_CHROME_RELIABLE_RANGE_ZOOM_FLAG.get()) {
const reliableRangeStart = await computeTraceReliableRangeStart(engine);
visibleStart = Time.max(visibleStart, reliableRangeStart);
}
// Move start of visible window to the first ftrace event
const ftraceBounds = await computeFtraceBounds(engine);
if (ftraceBounds !== null) {
// Avoid moving start of visible window past its end!
visibleStart = Time.min(ftraceBounds.start, visibleEnd);
}
return new TimeSpan(visibleStart, visibleEnd);
}
// TODO(sashwinbalaji): Move session UUID generation to TraceProcessor.
// computeGlobalUuid is a temporary measure to ensure multi-trace sessions
// have a unique cache key. This prevents collisions where a multi-trace
// session (e.g. a ZIP) would otherwise reuse the cache entry of its first
// component trace if that trace was previously opened individually.
async function computeGlobalUuid(uuids: string[]): Promise<string> {
if (uuids.length === 0) return '';
if (uuids.length === 1) return uuids[0];
const sortedUuids = [...uuids].sort();
return await sha1(sortedUuids.join(';'));
}
async function getTraceInfo(
engine: Engine,
app: AppImpl,
traceSource: TraceSource,
): Promise<TraceInfoImpl> {
const traceTime = await getTraceTimeBounds(engine);
// Find the first REALTIME or REALTIME_COARSE clock snapshot.
// Prioritize REALTIME over REALTIME_COARSE.
const query = `select
ts,
clock_value as clockValue,
clock_name as clockName
from clock_snapshot
where
snapshot_id = 0 AND
clock_name in ('REALTIME', 'REALTIME_COARSE')
`;
const result = await engine.query(query);
const it = result.iter({
ts: LONG,
clockValue: LONG,
clockName: STR,
});
let snapshot = {
clockName: '',
ts: Time.ZERO,
clockValue: Time.ZERO,
};
// Find the most suitable snapshot
for (let row = 0; it.valid(); it.next(), row++) {
if (it.clockName === 'REALTIME') {
snapshot = {
clockName: it.clockName,
ts: Time.fromRaw(it.ts),
clockValue: Time.fromRaw(it.clockValue),
};
break;
} else if (it.clockName === 'REALTIME_COARSE') {
if (snapshot.clockName !== 'REALTIME') {
snapshot = {
clockName: it.clockName,
ts: Time.fromRaw(it.ts),
clockValue: Time.fromRaw(it.clockValue),
};
}
}
}
// The max() is so the query returns NULL if the tz info doesn't exist.
const queryTz = `select max(int_value) as tzOffMin from metadata
where name = 'timezone_off_mins'`;
const resTz = await assertExists(engine).query(queryTz);
const tzOffMin = resTz.firstRow({tzOffMin: NUM_NULL}).tzOffMin ?? 0;
// This is the offset between the unix epoch and ts in the ts domain.
// I.e. the value of ts at the time of the unix epoch - usually some large
// negative value.
const unixOffset = Time.sub(snapshot.ts, snapshot.clockValue);
let traceTitle = '';
let traceUrl = '';
switch (traceSource.type) {
case 'FILE':
// Split on both \ and / (because C:\Windows\paths\are\like\this).
traceTitle = traceSource.file.name.split(/[/\\]/).pop()!;
const fileSizeMB = Math.ceil(traceSource.file.size / 1e6);
traceTitle += ` (${fileSizeMB} MB)`;
break;
case 'URL':
traceUrl = traceSource.url;
traceTitle = traceUrl.split('/').pop()!;
break;
case 'ARRAY_BUFFER':
traceTitle = traceSource.title;
traceUrl = traceSource.url ?? '';
const arrayBufferSizeMB = Math.ceil(traceSource.buffer.byteLength / 1e6);
traceTitle += ` (${arrayBufferSizeMB} MB)`;
break;
case 'HTTP_RPC':
traceTitle = `RPC @ ${HttpRpcEngine.hostAndPort}`;
break;
default:
break;
}
const traceTypes = await getTraceTypes(engine);
const hasFtrace =
(await engine.query(`select * from ftrace_event limit 1`)).numRows() > 0;
// Each trace in the session contributes to the global cache key. To maintain
// stable identifiers, we use the following priority:
// 1. Per-trace UUID: e.g. from a TraceUuid packet.
// 2. Global session UUID: ONLY used if no trace in the entire session has a
// specific UUID.
// 3. Trace ID + Type: e.g. '1-perf'. The last-resort fallback.
const uuidRes = await engine.query(`
INCLUDE PERFETTO MODULE std.traceinfo.trace;
SELECT DISTINCT
coalesce(
trace_uuid,
iif(
(SELECT COUNT(trace_uuid) FROM _metadata_by_trace) = 0,
extract_metadata('trace_uuid'),
NULL
),
trace_id || '-' || trace_type
) AS uuid
FROM _metadata_by_trace
`);
const uuids: string[] = [];
for (
const itUuid = uuidRes.iter({uuid: STR});
itUuid.valid();
itUuid.next()
) {
uuids.push(itUuid.uuid);
}
const uuid = await computeGlobalUuid(uuids);
updateStatus(app, 'Caching trace...');
const cached = await cacheTrace(traceSource, uuid);
const downloadable =
(traceSource.type === 'ARRAY_BUFFER' && !traceSource.localOnly) ||
traceSource.type === 'FILE' ||
traceSource.type === 'URL';
return {
...traceTime,
traceTitle,
traceUrl,
tzOffMin,
unixOffset,
importErrors: await getTraceErrors(engine),
source: traceSource,
traceTypes,
hasFtrace,
uuid,
cached,
downloadable,
};
}
async function getTraceTypes(engine: Engine): Promise<string[]> {
const result = await engine.query(`
INCLUDE PERFETTO MODULE std.traceinfo.trace;
select distinct trace_type as str_value
from _metadata_by_trace
`);
const traceTypes: string[] = [];
const it = result.iter({str_value: STR});
for (; it.valid(); it.next()) {
traceTypes.push(it.str_value);
}
return traceTypes;
}
async function getTraceTimeBounds(engine: Engine): Promise<TimeSpan> {
const result = await engine.query(
`select start_ts as startTs, end_ts as endTs from trace_bounds`,
);
const bounds = result.firstRow({
startTs: LONG,
endTs: LONG,
});
return new TimeSpan(Time.fromRaw(bounds.startTs), Time.fromRaw(bounds.endTs));
}
async function getTraceErrors(engine: Engine): Promise<number> {
const sql = `SELECT sum(value) as errs FROM stats WHERE severity != 'info'`;
const result = await engine.query(sql);
return result.firstRow({errs: NUM}).errs;
}
async function getTracingMetadataTimeBounds(engine: Engine): Promise<TimeSpan> {
const queryRes = await engine.query(`select
name,
int_value as intValue
from metadata
where name = 'tracing_started_ns' or name = 'tracing_disabled_ns'
or name = 'all_data_source_started_ns'`);
let startBound = Time.MIN;
let endBound = Time.MAX;
const it = queryRes.iter({name: STR, intValue: LONG_NULL});
for (; it.valid(); it.next()) {
const columnName = it.name;
const timestamp = it.intValue;
if (timestamp === null) continue;
if (columnName === 'tracing_disabled_ns') {
endBound = Time.min(endBound, Time.fromRaw(timestamp));
} else {
startBound = Time.max(startBound, Time.fromRaw(timestamp));
}
}
return new TimeSpan(startBound, endBound);
}