Source code

Revision control

Copy as Markdown

Other Tools

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
"use strict";
// Load the tracker very first in order to ensure tracking all objects created by DevTools.
// This is especially important for allocation sites. We need to catch the global the
// earliest possible in order to ensure that all allocation objects come with a stack.
//
// If we want to track DevTools module loader we should ensure loading Loader.sys.mjs within
// the `testScript` Function. i.e. after having calling startRecordingAllocations.
let tracker, releaseTrackerLoader;
{
const {
useDistinctSystemPrincipalLoader,
releaseDistinctSystemPrincipalLoader,
} = ChromeUtils.importESModule(
"resource://devtools/shared/loader/DistinctSystemPrincipalLoader.sys.mjs",
{ global: "shared" }
);
const requester = {};
const loader = useDistinctSystemPrincipalLoader(requester);
releaseTrackerLoader = () => releaseDistinctSystemPrincipalLoader(requester);
const { allocationTracker } = loader.require(
);
tracker = allocationTracker({ watchDevToolsGlobals: true });
}
// /!\ Be careful about imports/require
//
// Some tests may record the very first time we load a module.
// If we start loading them from here, we might only retrieve the already loaded
// module from the loader's cache. This would no longer highlight the cost
// of loading a new module from scratch.
//
// => Avoid loading devtools module as much as possible
// => If you really have to, lazy load them
ChromeUtils.defineLazyGetter(this, "TrackedObjects", () => {
return ChromeUtils.importESModule(
"resource://devtools/shared/test-helpers/tracked-objects.sys.mjs"
);
});
// So that PERFHERDER data can be extracted from the logs.
SimpleTest.requestCompleteLog();
// We have to disable testing mode, or various debug instructions are enabled.
// We especially want to disable redux store history, which would leak all the actions!
SpecialPowers.pushPrefEnv({
set: [["devtools.testing", false]],
});
// Set DEBUG_DEVTOOLS_ALLOCATIONS=allocations|leaks in order print debug informations.
const DEBUG_ALLOCATIONS = Services.env.get("DEBUG_DEVTOOLS_ALLOCATIONS");
async function addTab(url) {
const tab = BrowserTestUtils.addTab(gBrowser, url);
gBrowser.selectedTab = tab;
await BrowserTestUtils.browserLoaded(tab.linkedBrowser);
return tab;
}
/**
* This function will force some garbage collection before recording
* data about allocated objects.
*
* This accept an optional boolean to also record the content process objects
* of the current tab. That, in addition of objects from the parent process,
* which are always recorded.
*
* This return same data object which is meant to be passed to `stopRecordingAllocations` as-is.
*
* See README.md file in this folder.
*/
async function startRecordingAllocations({
alsoRecordContentProcess = false,
} = {}) {
// Also start recording allocations in the content process, if requested
if (alsoRecordContentProcess) {
await SpecialPowers.spawn(
gBrowser.selectedBrowser,
[DEBUG_ALLOCATIONS],
async debug_allocations => {
const { DevToolsLoader } = ChromeUtils.importESModule(
"resource://devtools/shared/loader/Loader.sys.mjs"
);
const {
useDistinctSystemPrincipalLoader,
releaseDistinctSystemPrincipalLoader,
} = ChromeUtils.importESModule(
"resource://devtools/shared/loader/DistinctSystemPrincipalLoader.sys.mjs"
);
const requester = {};
const loader = useDistinctSystemPrincipalLoader(requester);
const { allocationTracker } = loader.require(
);
// We watch all globals in the content process, (instead of only DevTools global in parent process)
// because we may easily leak web page objects, which aren't in DevTools global.
const tracker = allocationTracker({ watchAllGlobals: true });
// /!\ HACK: store tracker and releaseTrackerLoader on DevToolsLoader in order
// to be able to reuse them in a following call to SpecialPowers.spawn
DevToolsLoader.tracker = tracker;
DevToolsLoader.releaseTrackerLoader = () =>
releaseDistinctSystemPrincipalLoader(requester);
await tracker.startRecordingAllocations(debug_allocations);
}
);
// Trigger a GC in the parent process as this additional ContentTask
// seems to make harder to release objects created before we start recording.
await tracker.doGC();
}
await tracker.startRecordingAllocations(DEBUG_ALLOCATIONS);
}
/**
* See doc of startRecordingAllocations
*/
async function stopRecordingAllocations(
recordName,
{ alsoRecordContentProcess = false } = {}
) {
// Ensure that Memory API didn't ran out of buffers
ok(!tracker.overflowed, "Allocation were all recorded in the parent process");
// And finally, retrieve the record *after* having ran the test
const parentProcessData =
await tracker.stopRecordingAllocations(DEBUG_ALLOCATIONS);
const objectNodeIds = TrackedObjects.getAllNodeIds();
if (objectNodeIds.length) {
tracker.traceObjects(objectNodeIds);
}
let contentProcessData = null;
if (alsoRecordContentProcess) {
contentProcessData = await SpecialPowers.spawn(
gBrowser.selectedBrowser,
[DEBUG_ALLOCATIONS],
debug_allocations => {
const { DevToolsLoader } = ChromeUtils.importESModule(
"resource://devtools/shared/loader/Loader.sys.mjs"
);
const { tracker } = DevToolsLoader;
ok(
!tracker.overflowed,
"Allocation were all recorded in the content process"
);
return tracker.stopRecordingAllocations(debug_allocations);
}
);
}
const trackedObjectsInContent = await SpecialPowers.spawn(
gBrowser.selectedBrowser,
[],
() => {
const TrackedObjects = ChromeUtils.importESModule(
"resource://devtools/shared/test-helpers/tracked-objects.sys.mjs"
);
const objectNodeIds = TrackedObjects.getAllNodeIds();
if (objectNodeIds.length) {
const { DevToolsLoader } = ChromeUtils.importESModule(
"resource://devtools/shared/loader/Loader.sys.mjs"
);
const { tracker } = DevToolsLoader;
// Record the heap snapshot from the content process,
// and pass the record's filepath to the parent process
// As only the parent process can read the file because
// of sandbox restrictions made to content processes regarding file I/O.
const snapshotFile = tracker.getSnapshotFile();
return { snapshotFile, objectNodeIds };
}
return null;
}
);
if (trackedObjectsInContent) {
tracker.traceObjects(
trackedObjectsInContent.objectNodeIds,
trackedObjectsInContent.snapshotFile
);
}
// Craft the JSON object required to save data in talos database
info(
`The ${recordName} test leaked ${parentProcessData.objectsWithStack} objects (${parentProcessData.objectsWithoutStack} with missing allocation site) in the parent process`
);
const PERFHERDER_DATA = {
framework: {
name: "devtools",
},
suites: [
{
name: recordName + ":parent-process",
subtests: [
{
name: "objects-with-stacks",
value: parentProcessData.objectsWithStack,
},
{
name: "memory",
value: parentProcessData.memory,
},
],
},
],
};
if (alsoRecordContentProcess) {
info(
`The ${recordName} test leaked ${contentProcessData.objectsWithStack} objects (${contentProcessData.objectsWithoutStack} with missing allocation site) in the content process`
);
PERFHERDER_DATA.suites.push({
name: recordName + ":content-process",
subtests: [
{
name: "objects-with-stacks",
value: contentProcessData.objectsWithStack,
},
{
name: "memory",
value: contentProcessData.memory,
},
],
});
// Finally release the tracker loader in content process.
await SpecialPowers.spawn(gBrowser.selectedBrowser, [], () => {
const { DevToolsLoader } = ChromeUtils.importESModule(
"resource://devtools/shared/loader/Loader.sys.mjs"
);
DevToolsLoader.releaseTrackerLoader();
});
}
// And release the tracker loader in the parent process
releaseTrackerLoader();
// Log it to stdout so that perfherder can collect this data.
// This only works if we called `SimpleTest.requestCompleteLog()`!
info("PERFHERDER_DATA: " + JSON.stringify(PERFHERDER_DATA));
}