blob: 9b15d4d594e20bc29d602ec93b2b4bb35c5e1a16 [file] [log] [blame] [raw]
// Copyright (c) 2022, Compiler Explorer Authors
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// * Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
import $ from 'jquery';
import {Settings} from '../settings.js';
import {Chart, ChartData, defaults} from 'chart.js';
import 'chart.js/auto';
import {CompilationResult} from '../../types/compilation/compilation.interfaces.js';
import _ from 'underscore';
import {unwrap} from '../assert.js';
type Data = ChartData<'bar', number[], string> & {steps: number};
function pushTimingInfo(data: Data, step: string, time: number | string) {
if (typeof time === 'string') {
time = parseInt(time, 10);
}
data.labels?.push(`${step} (${Math.round(time * 100) / 100}ms)`);
data.datasets[0].data.push(time);
data.steps += time;
}
function concatTimings(data: Data, timings: {step: string; time: number}[]) {
for (const timing of timings) {
pushTimingInfo(data, timing.step, timing.time);
}
}
function addBuildResultToTimings(data: Data, buildResult: any) {
if (buildResult.packageDownloadAndUnzipTime) {
pushTimingInfo(data, 'Download binary from cache', buildResult.packageDownloadAndUnzipTime);
} else {
if (buildResult.downloads) {
concatTimings(data, buildResult.downloads);
}
if (buildResult.buildsteps) {
for (const step of buildResult.buildsteps) {
pushTimingInfo(data, step.step, step.execTime);
}
} else if (buildResult.execTime) {
pushTimingInfo(data, 'Compilation', buildResult.execTime);
}
}
}
function initializeChartDataFromResult(compileResult: CompilationResult, totalTime: number): Data {
const data: Data = {
steps: 0,
labels: [],
datasets: [
{
label: 'time in ms',
data: [],
borderWidth: 1,
barThickness: 20,
backgroundColor: ['red', 'orange', 'yellow', 'green', 'blue', 'indigo', 'violet'],
},
],
};
if (compileResult.retreivedFromCache) {
pushTimingInfo(data, 'Retrieve result from cache', unwrap(compileResult.retreivedFromCacheTime));
if (compileResult.packageDownloadAndUnzipTime) {
pushTimingInfo(data, 'Download binary from cache', unwrap(compileResult.execTime));
}
if (compileResult.execResult && compileResult.execResult.execTime) {
pushTimingInfo(data, 'Execution', compileResult.execResult.execTime);
}
} else {
addBuildResultToTimings(data, compileResult.buildResult || compileResult);
if (!compileResult.packageDownloadAndUnzipTime) {
if (compileResult.objdumpTime) {
pushTimingInfo(data, 'Disassembly', compileResult.objdumpTime);
}
if (compileResult.parsingTime) {
pushTimingInfo(data, 'ASM parsing', compileResult.parsingTime);
}
}
}
if (compileResult.didExecute) {
if (compileResult.execResult && compileResult.execResult.execTime) {
pushTimingInfo(data, 'Execution', compileResult.execResult.execTime);
} else {
pushTimingInfo(data, 'Execution', unwrap(compileResult.execTime));
}
}
if (compileResult.processExecutionResultTime !== undefined) {
pushTimingInfo(data, 'Process execution result', compileResult.processExecutionResultTime);
}
if (compileResult.hasLLVMOptPipelineOutput && !_.isString(compileResult.llvmOptPipelineOutput)) {
if (compileResult.llvmOptPipelineOutput?.clangTime !== undefined) {
pushTimingInfo(data, 'Llvm opt pipeline clang time', compileResult.llvmOptPipelineOutput.clangTime);
}
if (compileResult.llvmOptPipelineOutput?.parseTime !== undefined) {
pushTimingInfo(data, 'Llvm opt pipeline parse time', compileResult.llvmOptPipelineOutput.parseTime);
}
}
const stepsTotal = data.steps;
pushTimingInfo(data, 'Network, JS, waiting, etc.', totalTime - stepsTotal);
if (totalTime - stepsTotal < 0) {
data.datasets[0].data = [totalTime];
data.labels = ['Browser cache'];
}
return data;
}
function displayData(data: Data) {
const modal = $('#timing-info');
const chartDiv = modal.find('#chart');
chartDiv.html('');
// eslint thinks "This assertion is unnecessary since it does not change the type of the expression"
// Typescript disagrees.
// eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion
const canvas = $('<canvas id="timing-chart" width="400" height="400"></canvas>') as JQuery<HTMLCanvasElement>;
chartDiv.append(canvas);
const settings = Settings.getStoredSettings();
let fontColour = defaults.color.toString();
if (settings.theme !== 'default') {
fontColour = '#ffffff';
}
new Chart(canvas, {
type: 'bar',
data: data,
options: {
scales: {
xAxis: {
beginAtZero: true,
grid: {
color: fontColour,
tickColor: fontColour,
},
ticks: {color: fontColour},
},
yAxis: {
beginAtZero: true,
grid: {
color: fontColour,
tickColor: fontColour,
},
ticks: {color: fontColour},
},
},
plugins: {
legend: {
labels: {
color: fontColour,
font: {size: 18},
},
},
},
},
});
modal.modal('show');
}
export function displayCompilationTiming(compileResult: CompilationResult | null, totalTime: number) {
if (compileResult) {
const data = initializeChartDataFromResult(compileResult, totalTime);
displayData(data);
}
}