blob: 39203b422a0e3503edf7d5f5da5051d1cadb333e [file] [log] [blame]
/*
* Copyright (C) 2016 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. 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 APPLE INC. AND ITS 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 APPLE INC. OR ITS 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.
*/
WI.CallingContextTree = class CallingContextTree
{
constructor(target, type)
{
this._target = target;
this._type = type || WI.CallingContextTree.Type.TopDown;
const sourceID = -1;
const line = -1;
const column = -1;
const name = "<root>";
const url = null;
this._root = new WI.CallingContextTreeNode(sourceID, line, column, name, url);
this._totalNumberOfSamples = 0;
}
// Public
get target() { return this._target; }
get type() { return this._type; }
get totalNumberOfSamples() { return this._totalNumberOfSamples; }
totalDurationInTimeRange(startTime, endTime)
{
return this._root.filteredTimestampsAndDuration(startTime, endTime).duration;
}
updateTreeWithStackTrace({timestamp, stackFrames}, duration)
{
this._totalNumberOfSamples++;
let node = this._root;
node.addTimestampAndExpressionLocation(timestamp, duration, null);
switch (this._type) {
case WI.CallingContextTree.Type.TopDown:
for (let i = stackFrames.length; i--; ) {
let stackFrame = stackFrames[i];
node = node.findOrMakeChild(stackFrame);
node.addTimestampAndExpressionLocation(timestamp, duration, stackFrame.expressionLocation || null, i === 0);
}
break;
case WI.CallingContextTree.Type.BottomUp:
for (let i = 0; i < stackFrames.length; ++i) {
let stackFrame = stackFrames[i];
node = node.findOrMakeChild(stackFrame);
node.addTimestampAndExpressionLocation(timestamp, duration, stackFrame.expressionLocation || null, i === 0);
}
break;
case WI.CallingContextTree.Type.TopFunctionsTopDown:
for (let i = stackFrames.length; i--; ) {
node = this._root;
for (let j = i + 1; j--; ) {
let stackFrame = stackFrames[j];
node = node.findOrMakeChild(stackFrame);
node.addTimestampAndExpressionLocation(timestamp, duration, stackFrame.expressionLocation || null, j === 0);
}
}
break;
case WI.CallingContextTree.Type.TopFunctionsBottomUp:
for (let i = 0; i < stackFrames.length; i++) {
node = this._root;
for (let j = i; j < stackFrames.length; j++) {
let stackFrame = stackFrames[j];
node = node.findOrMakeChild(stackFrame);
node.addTimestampAndExpressionLocation(timestamp, duration, stackFrame.expressionLocation || null, j === 0);
}
}
break;
default:
console.assert(false, "This should not be reached.");
break;
}
}
toCPUProfilePayload(startTime, endTime)
{
let cpuProfile = {};
let roots = [];
let numSamplesInTimeRange = this._root.filteredTimestampsAndDuration(startTime, endTime).timestamps.length;
this._root.forEachChild((child) => {
if (child.hasStackTraceInTimeRange(startTime, endTime))
roots.push(child.toCPUProfileNode(numSamplesInTimeRange, startTime, endTime));
});
cpuProfile.rootNodes = roots;
return cpuProfile;
}
forEachChild(callback)
{
this._root.forEachChild(callback);
}
forEachNode(callback)
{
this._root.forEachNode(callback);
}
};
WI.CallingContextTree.Type = {
TopDown: Symbol("TopDown"),
BottomUp: Symbol("BottomUp"),
TopFunctionsTopDown: Symbol("TopFunctionsTopDown"),
TopFunctionsBottomUp: Symbol("TopFunctionsBottomUp"),
};