blob: 2a5b787492fb2d7d4a68f8e75c15ee844bce684c [file] [log] [blame]
// 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 {Actions} from '../common/actions';
import {TraceSource} from '../common/state';
import * as trace_to_text from '../gen/trace_to_text';
import {globals} from './globals';
export function ConvertTrace(trace: Blob, truncate?: 'start'|'end') {
const mod = trace_to_text({
noInitialRun: true,
locateFile: (s: string) => s,
print: updateStatus,
printErr: updateStatus,
onRuntimeInitialized: () => {
updateStatus('Converting trace');
const outPath = '/trace.json';
if (truncate === undefined) {
mod.callMain(['json', '/fs/trace.proto', outPath]);
} else {
mod.callMain(
['json', '--truncate', truncate, '/fs/trace.proto', outPath]);
}
updateStatus('Trace conversion completed');
const fsNode = mod.FS.lookupPath(outPath).node;
const data = fsNode.contents.buffer;
const size = fsNode.usedBytes;
globals.publish('LegacyTrace', {data, size}, /*transfer=*/[data]);
mod.FS.unlink(outPath);
},
onAbort: () => {
console.log('ABORT');
},
});
mod.FS.mkdir('/fs');
mod.FS.mount(
mod.FS.filesystems.WORKERFS,
{blobs: [{name: 'trace.proto', data: trace}]},
'/fs');
// TODO removeme.
(self as {} as {mod: {}}).mod = mod;
}
export async function ConvertTraceToPprof(
pid: number, src: TraceSource, ts1: number, ts2?: number) {
generateBlob(src).then(result => {
const mod = trace_to_text({
noInitialRun: true,
locateFile: (s: string) => s,
print: updateStatus,
printErr: updateStatus,
onRuntimeInitialized: () => {
updateStatus('Converting trace');
const timestamps = `${ts1}${ts2 === undefined ? '' : `,${ts2}`}`;
mod.callMain([
'profile',
`--pid`,
`${pid}`,
`--timestamps`,
timestamps,
'/fs/trace.proto'
]);
updateStatus('Trace conversion completed');
const heapDirName =
Object.keys(mod.FS.lookupPath('/tmp/').node.contents)[0];
const heapDirContents =
mod.FS.lookupPath(`/tmp/${heapDirName}`).node.contents;
const heapDumpFiles = Object.keys(heapDirContents);
let fileNum = 0;
heapDumpFiles.forEach(heapDump => {
const fileContents =
mod.FS.lookupPath(`/tmp/${heapDirName}/${heapDump}`)
.node.contents;
fileNum++;
const fileName = `/heap_dump.${fileNum}.${pid}.pb`;
downloadFile(new Blob([fileContents]), fileName);
});
updateStatus('Profile(s) downloaded');
},
onAbort: () => {
console.log('ABORT');
},
});
mod.FS.mkdir('/fs');
mod.FS.mount(
mod.FS.filesystems.WORKERFS,
{blobs: [{name: 'trace.proto', data: result}]},
'/fs');
});
}
async function generateBlob(src: TraceSource) {
let blob: Blob = new Blob();
if (src.type === 'URL') {
const resp = await fetch(src.url);
if (resp.status !== 200) {
throw new Error(`fetch() failed with HTTP error ${resp.status}`);
}
blob = await resp.blob();
} else if (src.type === 'ARRAY_BUFFER') {
blob = new Blob([new Uint8Array(src.buffer, 0, src.buffer.byteLength)]);
} else if (src.type === 'FILE') {
blob = src.file;
} else {
throw new Error(`Conversion not supported for ${JSON.stringify(src)}`);
}
return blob;
}
function downloadFile(file: Blob, name: string) {
globals.publish('FileDownload', {file, name});
}
function updateStatus(msg: {}) {
console.log(msg);
globals.dispatch(Actions.updateStatus({
msg: msg.toString(),
timestamp: Date.now() / 1000,
}));
}