summaryrefslogtreecommitdiffstats
path: root/toolkit/components/aboutperformance
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-28 14:29:10 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-28 14:29:10 +0000
commit2aa4a82499d4becd2284cdb482213d541b8804dd (patch)
treeb80bf8bf13c3766139fbacc530efd0dd9d54394c /toolkit/components/aboutperformance
parentInitial commit. (diff)
downloadfirefox-2aa4a82499d4becd2284cdb482213d541b8804dd.tar.xz
firefox-2aa4a82499d4becd2284cdb482213d541b8804dd.zip
Adding upstream version 86.0.1.upstream/86.0.1upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'toolkit/components/aboutperformance')
-rw-r--r--toolkit/components/aboutperformance/content/aboutPerformance.css242
-rw-r--r--toolkit/components/aboutperformance/content/aboutPerformance.html29
-rw-r--r--toolkit/components/aboutperformance/content/aboutPerformance.js1049
-rw-r--r--toolkit/components/aboutperformance/jar.mn8
-rw-r--r--toolkit/components/aboutperformance/moz.build12
-rw-r--r--toolkit/components/aboutperformance/tests/browser/browser.ini7
-rw-r--r--toolkit/components/aboutperformance/tests/browser/browser_aboutperformance.js106
-rw-r--r--toolkit/components/aboutperformance/tests/browser/browser_compartments.html21
-rw-r--r--toolkit/components/aboutperformance/tests/browser/browser_compartments_frame.html13
-rw-r--r--toolkit/components/aboutperformance/tests/browser/browser_compartments_script.js11
10 files changed, 1498 insertions, 0 deletions
diff --git a/toolkit/components/aboutperformance/content/aboutPerformance.css b/toolkit/components/aboutperformance/content/aboutPerformance.css
new file mode 100644
index 0000000000..42275f1da0
--- /dev/null
+++ b/toolkit/components/aboutperformance/content/aboutPerformance.css
@@ -0,0 +1,242 @@
+/* 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/. */
+
+@import url("chrome://global/skin/in-content/common.css");
+
+html {
+ background-color: var(--in-content-page-background);
+}
+body {
+ overflow-x: hidden;
+}
+#dispatch-table {
+ user-select: none;
+ font-size: 1em;
+ border-spacing: 0;
+ background-color: var(--in-content-box-background);
+ margin: 0;
+ position: absolute;
+ top: 0;
+ inset-inline-start: 0;
+ width: 100%;
+ height: 100%;
+ min-width: 40em;
+}
+
+/* Avoid scrolling the header */
+#dispatch-tbody {
+ display: block;
+ margin-top: 2em;
+}
+#dispatch-thead {
+ position: fixed;
+ z-index: 1;
+ height: 2em;
+ border-bottom: 1px solid var(--in-content-border-color);
+ min-width: 40em;
+ background-color: var(--in-content-box-background);
+}
+tr {
+ display: table;
+ table-layout: fixed;
+ width: 100%;
+}
+td:nth-child(2) {
+ width: 8em;
+}
+td:nth-child(3) {
+ width: 12em;
+}
+td:nth-child(4) {
+ width: 5em;
+}
+#dispatch-tbody td:nth-child(4) {
+ text-align: end;
+}
+td:nth-child(5) {
+ width: 20px;
+}
+
+/* Show action icons on selected or hovered rows */
+tr:is([selected], :hover) > td > .action-icon {
+ padding: 1px 10px;
+ opacity: 1;
+}
+/* The action icons have a relative position, so that we can use
+ * absolutely positioned ::before and ::after pseudo elements.
+ * ::before is used to display the square background on hover/active
+ * ::after is used to display the icons as a background that can be
+ * flipped using a CSS transform in RTL mode. */
+.action-icon {
+ position: relative;
+ opacity: 0;
+}
+/* Ensure both pseudo elements have the same size and position. */
+.action-icon::before, .action-icon::after {
+ height: 200%;
+ position: absolute;
+ top: -50%;
+ inset-inline-start: -3px;
+ padding-inline: 13px;
+}
+
+/* square background */
+.action-icon::before {
+ content: "";
+ background-color: currentColor;
+ opacity: 0;
+}
+.action-icon:hover::before {
+ opacity: 0.1;
+}
+.action-icon:hover:active::before {
+ opacity: 0.2;
+}
+
+/* icons */
+.action-icon::after {
+ -moz-context-properties: fill, fill-opacity;
+ fill: currentColor;
+ fill-opacity: 0;
+ background-repeat: no-repeat;
+ background-position: center;
+ line-height: 100%;
+}
+.addon-icon::after {
+ content: url("chrome://global/skin/icons/shortcut.svg");
+ top: 2px;
+ inset-inline-start: -11px;
+ width: 16px;
+}
+.addon-icon:dir(rtl)::after {
+ transform: scaleX(-1);
+}
+.close-icon::after {
+ content: url("chrome://global/skin/icons/close.svg");
+ height: 100%;
+ top: 0;
+ inset-inline-start: -13px;
+ transform: scale(1.2);
+}
+
+#dispatch-thead > tr {
+ height: inherit;
+}
+
+#dispatch-thead > tr > td {
+ border: none;
+ background-color: var(--in-content-button-background);
+}
+#dispatch-thead > tr > td:not(:first-child) {
+ border-inline-start-width: 1px;
+ border-inline-start-style: solid;
+ border-image: linear-gradient(transparent 0%, transparent 20%, var(--in-content-box-border-color) 20%, var(--in-content-box-border-color) 80%, transparent 80%, transparent 100%) 1 1;
+ border-bottom: 1px solid var(--in-content-border-color);
+}
+td {
+ padding: 5px 10px;
+ min-height: 2em;
+ color: var(--in-content-text-color);
+ max-width: 70vw;
+ overflow: hidden;
+ white-space: nowrap;
+}
+#dispatch-tbody > tr > td:first-child {
+ text-overflow: ellipsis;
+ padding-inline-start: 32px;
+ background-repeat: no-repeat;
+ background-size: 16px 16px;
+ background-position-y: center;
+ -moz-context-properties: fill;
+ fill: currentColor;
+}
+#dispatch-tbody > tr > td.root {
+ background-position-x: left 36px;
+ padding-inline-start: 62px;
+}
+#dispatch-tbody > tr > td.root:dir(rtl) {
+ background-position-x: right 36px;
+}
+.twisty {
+ margin-inline: -62px 26px;
+ padding-inline: 18px;
+ position: relative;
+}
+/* Putting the background image in a positioned pseudo element lets us
+ * use CSS transforms on the background image, which we need for rtl. */
+.twisty::before {
+ content: url("chrome://global/skin/icons/twisty-collapsed.svg");
+ position: absolute;
+ display: block;
+ line-height: 50%;
+ top: 4px; /* Half the image's height */
+ width: 100%;
+ inset-inline-start: 0;
+ text-align: center;
+ -moz-context-properties: fill;
+ fill: currentColor;
+}
+.twisty:dir(rtl)::before {
+ transform: scaleX(-1);
+}
+.twisty.open::before {
+ content: url("chrome://global/skin/icons/twisty-expanded.svg");
+}
+#dispatch-tbody > tr > td.indent {
+ padding-inline-start: 88px;
+ background-position-x: left 62px;
+}
+#dispatch-tbody > tr > td.indent:dir(rtl) {
+ background-position-x: right 62px;
+}
+#dispatch-tbody > tr > td.tracker {
+ background-image: url("chrome://browser/skin/controlcenter/trackers.svg");
+ -moz-context-properties: fill;
+ fill: rgb(224, 41, 29);
+}
+#dispatch-tbody > tr > td.worker {
+ background-image: url("chrome://devtools/skin/images/debugging-workers.svg");
+ -moz-context-properties: fill;
+ fill: #808080;
+}
+
+#dispatch-tbody > tr[selected] > td {
+ background-color: var(--in-content-item-selected);
+ color: var(--in-content-selected-text);
+}
+#dispatch-tbody > tr:hover {
+ background-color: var(--in-content-item-hover);
+}
+
+.clickable {
+ background-repeat: no-repeat;
+ background-position: right 4px center;
+}
+.clickable:dir(rtl) {
+ background-position-x: left 4px;
+}
+.asc {
+ background-image: url(chrome://global/skin/icons/arrow-up-12.svg);
+ -moz-context-properties: fill;
+ fill: currentColor;
+}
+.desc {
+ background-image: url(chrome://global/skin/icons/arrow-dropdown-12.svg);
+ -moz-context-properties: fill;
+ fill: currentColor;
+}
+#dispatch-thead > tr > td.clickable:hover {
+ background-color: var(--in-content-button-background-hover);
+}
+#dispatch-thead > tr > td.clickable:active {
+ background-color: var(--in-content-button-background-active);
+}
+
+.energy-impact {
+ --bar-width: 0;
+ background: linear-gradient(to right, var(--blue-40) calc(var(--bar-width) * 1%), transparent calc(var(--bar-width) * 1%));
+}
+.energy-impact:dir(rtl) {
+ background: linear-gradient(to left, var(--blue-40) calc(var(--bar-width) * 1%), transparent calc(var(--bar-width) * 1%));
+}
diff --git a/toolkit/components/aboutperformance/content/aboutPerformance.html b/toolkit/components/aboutperformance/content/aboutPerformance.html
new file mode 100644
index 0000000000..a6d907d74b
--- /dev/null
+++ b/toolkit/components/aboutperformance/content/aboutPerformance.html
@@ -0,0 +1,29 @@
+<!-- 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/. -->
+<!DOCTYPE html>
+<html>
+ <head>
+ <meta http-equiv="Content-Security-Policy" content="default-src chrome:;img-src data:; object-src 'none'">
+ <title data-l10n-id="about-performance-title"></title>
+ <link rel="icon" id="favicon" href="chrome://global/skin/icons/performance.svg">
+ <link rel="stylesheet" href="chrome://global/skin/in-content/common.css">
+ <link rel="localization" href="toolkit/about/aboutPerformance.ftl">
+ <script src="chrome://global/content/aboutPerformance.js"></script>
+ <link rel="stylesheet" href="chrome://global/content/aboutPerformance.css">
+ </head>
+ <body>
+ <table id="dispatch-table">
+ <thead id="dispatch-thead">
+ <tr>
+ <td class="clickable" id="column-name" data-l10n-id="column-name"></td>
+ <td class="clickable" id="column-type" data-l10n-id="column-type"></td>
+ <td class="clickable" id="column-energy-impact" data-l10n-id="column-energy-impact"></td>
+ <td class="clickable" id="column-memory" data-l10n-id="column-memory"></td>
+ <td></td><!-- actions -->
+ </tr>
+ </thead>
+ <tbody id="dispatch-tbody"></tbody>
+ </table>
+ </body>
+</html>
diff --git a/toolkit/components/aboutperformance/content/aboutPerformance.js b/toolkit/components/aboutperformance/content/aboutPerformance.js
new file mode 100644
index 0000000000..a243a6942b
--- /dev/null
+++ b/toolkit/components/aboutperformance/content/aboutPerformance.js
@@ -0,0 +1,1049 @@
+/* -*- indent-tabs-mode: nil; js-indent-level: 2 -*-*/
+/* vim: set ts=8 sts=2 et sw=2 tw=80: */
+/* 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";
+
+const { Services } = ChromeUtils.import("resource://gre/modules/Services.jsm");
+const { AddonManager } = ChromeUtils.import(
+ "resource://gre/modules/AddonManager.jsm"
+);
+const { ExtensionParent } = ChromeUtils.import(
+ "resource://gre/modules/ExtensionParent.jsm"
+);
+
+const { WebExtensionPolicy } = Cu.getGlobalForObject(Services);
+
+// Time in ms before we start changing the sort order again after receiving a
+// mousemove event.
+const TIME_BEFORE_SORTING_AGAIN = 5000;
+
+// How often we should add a sample to our buffer.
+const BUFFER_SAMPLING_RATE_MS = 1000;
+
+// The age of the oldest sample to keep.
+const BUFFER_DURATION_MS = 10000;
+
+// How often we should update
+const UPDATE_INTERVAL_MS = 2000;
+
+// The name of the application
+const BRAND_BUNDLE = Services.strings.createBundle(
+ "chrome://branding/locale/brand.properties"
+);
+const BRAND_NAME = BRAND_BUNDLE.GetStringFromName("brandShortName");
+
+function extensionCountersEnabled() {
+ return Services.prefs.getBoolPref(
+ "extensions.webextensions.enablePerformanceCounters",
+ false
+ );
+}
+
+// The ids of system add-ons, so that we can hide them when the
+// toolkit.aboutPerformance.showInternals pref is false.
+// The API to access addons is async, so we cache the list during init.
+// The list is unlikely to change while the about:performance
+// tab is open, so not updating seems fine.
+var gSystemAddonIds = new Set();
+
+let tabFinder = {
+ update() {
+ this._map = new Map();
+ for (let win of Services.wm.getEnumerator("navigator:browser")) {
+ let tabbrowser = win.gBrowser;
+ for (let browser of tabbrowser.browsers) {
+ let id = browser.outerWindowID; // May be `null` if the browser isn't loaded yet
+ if (id != null) {
+ this._map.set(id, browser);
+ }
+ }
+ if (tabbrowser.preloadedBrowser) {
+ let browser = tabbrowser.preloadedBrowser;
+ if (browser.outerWindowID) {
+ this._map.set(browser.outerWindowID, browser);
+ }
+ }
+ }
+ },
+
+ /**
+ * Find the <xul:tab> for a window id.
+ *
+ * This is useful e.g. for reloading or closing tabs.
+ *
+ * @return null If the xul:tab could not be found, e.g. if the
+ * windowId is that of a chrome window.
+ * @return {{tabbrowser: <xul:tabbrowser>, tab: <xul.tab>}} The
+ * tabbrowser and tab if the latter could be found.
+ */
+ get(id) {
+ let browser = this._map.get(id);
+ if (!browser) {
+ return null;
+ }
+ let tabbrowser = browser.getTabBrowser();
+ if (!tabbrowser) {
+ return {
+ tabbrowser: null,
+ tab: {
+ getAttribute() {
+ return "";
+ },
+ linkedBrowser: browser,
+ },
+ };
+ }
+ return { tabbrowser, tab: tabbrowser.getTabForBrowser(browser) };
+ },
+
+ getAny(ids) {
+ for (let id of ids) {
+ let result = this.get(id);
+ if (result) {
+ return result;
+ }
+ }
+ return null;
+ },
+};
+
+/**
+ * Returns a Promise that's resolved after the next turn of the event loop.
+ *
+ * Just returning a resolved Promise would mean that any `then` callbacks
+ * would be called right after the end of the current turn, so `setTimeout`
+ * is used to delay Promise resolution until the next turn.
+ *
+ * In mochi tests, it's possible for this to be called after the
+ * about:performance window has been torn down, which causes `setTimeout` to
+ * throw an NS_ERROR_NOT_INITIALIZED exception. In that case, returning
+ * `undefined` is fine.
+ */
+function wait(ms = 0) {
+ try {
+ let resolve;
+ let p = new Promise(resolve_ => {
+ resolve = resolve_;
+ });
+ setTimeout(resolve, ms);
+ return p;
+ } catch (e) {
+ dump(
+ "WARNING: wait aborted because of an invalid Window state in aboutPerformance.js.\n"
+ );
+ return undefined;
+ }
+}
+
+/**
+ * Utilities for dealing with state
+ */
+var State = {
+ /**
+ * Indexed by the number of minutes since the snapshot was taken.
+ *
+ * @type {Array<ApplicationSnapshot>}
+ */
+ _buffer: [],
+ /**
+ * The latest snapshot.
+ *
+ * @type ApplicationSnapshot
+ */
+ _latest: null,
+
+ async _promiseSnapshot() {
+ let addons = WebExtensionPolicy.getActiveExtensions();
+ let addonHosts = new Map();
+ for (let addon of addons) {
+ addonHosts.set(addon.mozExtensionHostname, addon.id);
+ }
+
+ let counters = await ChromeUtils.requestPerformanceMetrics();
+ let tabs = {};
+ for (let counter of counters) {
+ let {
+ items,
+ host,
+ pid,
+ counterId,
+ windowId,
+ duration,
+ isWorker,
+ memoryInfo,
+ isTopLevel,
+ } = counter;
+ // If a worker has a windowId of 0 or max uint64, attach it to the
+ // browser UI (doc group with id 1).
+ if (isWorker && (windowId == 18446744073709552000 || !windowId)) {
+ windowId = 1;
+ }
+ let dispatchCount = 0;
+ for (let { count } of items) {
+ dispatchCount += count;
+ }
+
+ let memory = 0;
+ for (let field in memoryInfo) {
+ if (field == "media") {
+ for (let mediaField of ["audioSize", "videoSize", "resourcesSize"]) {
+ memory += memoryInfo.media[mediaField];
+ }
+ continue;
+ }
+ memory += memoryInfo[field];
+ }
+
+ let tab;
+ let id = windowId;
+ if (addonHosts.has(host)) {
+ id = addonHosts.get(host);
+ }
+ if (id in tabs) {
+ tab = tabs[id];
+ } else {
+ tab = {
+ windowId,
+ host,
+ dispatchCount: 0,
+ duration: 0,
+ memory: 0,
+ children: [],
+ };
+ tabs[id] = tab;
+ }
+ tab.dispatchCount += dispatchCount;
+ tab.duration += duration;
+ tab.memory += memory;
+ if (!isTopLevel || isWorker) {
+ tab.children.push({
+ host,
+ isWorker,
+ dispatchCount,
+ duration,
+ memory,
+ counterId: pid + ":" + counterId,
+ });
+ }
+ }
+
+ if (extensionCountersEnabled()) {
+ let extCounters = await ExtensionParent.ParentAPIManager.retrievePerformanceCounters();
+ for (let [id, apiMap] of extCounters) {
+ let dispatchCount = 0,
+ duration = 0;
+ for (let [, counter] of apiMap) {
+ dispatchCount += counter.calls;
+ duration += counter.duration;
+ }
+
+ let tab;
+ if (id in tabs) {
+ tab = tabs[id];
+ } else {
+ tab = {
+ windowId: 0,
+ host: id,
+ dispatchCount: 0,
+ duration: 0,
+ memory: 0,
+ children: [],
+ };
+ tabs[id] = tab;
+ }
+ tab.dispatchCount += dispatchCount;
+ tab.duration += duration;
+ }
+ }
+
+ return { tabs, date: Cu.now() };
+ },
+
+ /**
+ * Update the internal state.
+ *
+ * @return {Promise}
+ */
+ async update() {
+ // If the buffer is empty, add one value for bootstraping purposes.
+ if (!this._buffer.length) {
+ this._latest = await this._promiseSnapshot();
+ this._buffer.push(this._latest);
+ await wait(BUFFER_SAMPLING_RATE_MS * 1.1);
+ }
+
+ let now = Cu.now();
+
+ // If we haven't sampled in a while, add a sample to the buffer.
+ let latestInBuffer = this._buffer[this._buffer.length - 1];
+ let deltaT = now - latestInBuffer.date;
+ if (deltaT > BUFFER_SAMPLING_RATE_MS) {
+ this._latest = await this._promiseSnapshot();
+ this._buffer.push(this._latest);
+ }
+
+ // If we have too many samples, remove the oldest sample.
+ let oldestInBuffer = this._buffer[0];
+ if (oldestInBuffer.date + BUFFER_DURATION_MS < this._latest.date) {
+ this._buffer.shift();
+ }
+ },
+
+ // We can only know asynchronously if an origin is matched by the tracking
+ // protection list, so we cache the result for faster future lookups.
+ _trackingState: new Map(),
+ isTracker(host) {
+ if (!this._trackingState.has(host)) {
+ // Temporarily set to false to avoid doing several lookups if a site has
+ // several subframes on the same domain.
+ this._trackingState.set(host, false);
+ if (host.startsWith("about:") || host.startsWith("moz-nullprincipal")) {
+ return false;
+ }
+
+ let uri = Services.io.newURI("http://" + host);
+ let classifier = Cc["@mozilla.org/url-classifier/dbservice;1"].getService(
+ Ci.nsIURIClassifier
+ );
+ let feature = classifier.getFeatureByName("tracking-protection");
+ if (!feature) {
+ return false;
+ }
+
+ classifier.asyncClassifyLocalWithFeatures(
+ uri,
+ [feature],
+ Ci.nsIUrlClassifierFeature.blocklist,
+ list => {
+ if (list.length) {
+ this._trackingState.set(host, true);
+ }
+ }
+ );
+ }
+ return this._trackingState.get(host);
+ },
+
+ getCounters() {
+ tabFinder.update();
+ // We rebuild the maps during each iteration to make sure that
+ // we do not maintain references to groups that has been removed
+ // (e.g. pages that have been closed).
+
+ let previous = this._buffer[Math.max(this._buffer.length - 2, 0)].tabs;
+ let current = this._latest.tabs;
+ let counters = [];
+ for (let id of Object.keys(current)) {
+ let tab = current[id];
+ let oldest;
+ for (let index = 0; index <= this._buffer.length - 2; ++index) {
+ if (id in this._buffer[index].tabs) {
+ oldest = this._buffer[index].tabs[id];
+ break;
+ }
+ }
+ let prev = previous[id];
+ let host = tab.host;
+
+ let type = "other";
+ let name = `${host} (${id})`;
+ let image = "chrome://mozapps/skin/places/defaultFavicon.svg";
+ let found = tabFinder.get(parseInt(id));
+ if (found) {
+ if (found.tabbrowser) {
+ name = found.tab.getAttribute("label");
+ image = found.tab.getAttribute("image");
+ type = "tab";
+ } else {
+ name = {
+ id: "preloaded-tab",
+ title: found.tab.linkedBrowser.contentTitle,
+ };
+ }
+ } else if (id == 1) {
+ name = BRAND_NAME;
+ image = "chrome://branding/content/icon32.png";
+ type = "browser";
+ } else if (/^[a-f0-9]{8}(-[a-f0-9]{4}){3}-[a-f0-9]{12}$/.test(host)) {
+ let addon = WebExtensionPolicy.getByHostname(host);
+ if (!addon) {
+ continue;
+ }
+ name = `${addon.name} (${addon.id})`;
+ image = "chrome://mozapps/skin/extensions/extension.svg";
+ type = gSystemAddonIds.has(addon.id) ? "system-addon" : "addon";
+ } else if (id == 0 && !tab.isWorker) {
+ name = { id: "ghost-windows" };
+ }
+
+ if (
+ type != "tab" &&
+ type != "addon" &&
+ !Services.prefs.getBoolPref(
+ "toolkit.aboutPerformance.showInternals",
+ false
+ )
+ ) {
+ continue;
+ }
+
+ // Create a map of all the child items from the previous time we read the
+ // counters, indexed by counterId so that we can quickly find the previous
+ // value for any subitem.
+ let prevChildren = new Map();
+ if (prev) {
+ for (let child of prev.children) {
+ prevChildren.set(child.counterId, child);
+ }
+ }
+ // For each subitem, create a new object including the deltas since the previous time.
+ let children = tab.children.map(child => {
+ let {
+ host,
+ dispatchCount,
+ duration,
+ memory,
+ isWorker,
+ counterId,
+ } = child;
+ let dispatchesSincePrevious = dispatchCount;
+ let durationSincePrevious = duration;
+ if (prevChildren.has(counterId)) {
+ let prevCounter = prevChildren.get(counterId);
+ dispatchesSincePrevious -= prevCounter.dispatchCount;
+ durationSincePrevious -= prevCounter.duration;
+ prevChildren.delete(counterId);
+ }
+
+ return {
+ host,
+ dispatchCount,
+ duration,
+ isWorker,
+ memory,
+ dispatchesSincePrevious,
+ durationSincePrevious,
+ };
+ });
+
+ // Any item that remains in prevChildren is a subitem that no longer
+ // exists in the current sample; remember the values of its counters
+ // so that the values don't go down for the parent item.
+ tab.dispatchesFromFormerChildren =
+ (prev && prev.dispatchesFromFormerChildren) || 0;
+ tab.durationFromFormerChildren =
+ (prev && prev.durationFromFormerChildren) || 0;
+ for (let [, counter] of prevChildren) {
+ tab.dispatchesFromFormerChildren += counter.dispatchCount;
+ tab.durationFromFormerChildren += counter.duration;
+ }
+
+ // Create the object representing the counters of the parent item including
+ // the deltas from the previous times.
+ let dispatches = tab.dispatchCount + tab.dispatchesFromFormerChildren;
+ let duration = tab.duration + tab.durationFromFormerChildren;
+ let durationSincePrevious = NaN;
+ let dispatchesSincePrevious = NaN;
+ let dispatchesSinceStartOfBuffer = NaN;
+ let durationSinceStartOfBuffer = NaN;
+ if (prev) {
+ durationSincePrevious =
+ duration - prev.duration - (prev.durationFromFormerChildren || 0);
+ dispatchesSincePrevious =
+ dispatches -
+ prev.dispatchCount -
+ (prev.dispatchesFromFormerChildren || 0);
+ }
+ if (oldest) {
+ dispatchesSinceStartOfBuffer =
+ dispatches -
+ oldest.dispatchCount -
+ (oldest.dispatchesFromFormerChildren || 0);
+ durationSinceStartOfBuffer =
+ duration - oldest.duration - (oldest.durationFromFormerChildren || 0);
+ }
+ counters.push({
+ id,
+ name,
+ image,
+ type,
+ memory: tab.memory,
+ totalDispatches: dispatches,
+ totalDuration: duration,
+ durationSincePrevious,
+ dispatchesSincePrevious,
+ durationSinceStartOfBuffer,
+ dispatchesSinceStartOfBuffer,
+ children,
+ });
+ }
+ return counters;
+ },
+
+ getMaxEnergyImpact(counters) {
+ return Math.max(
+ ...counters.map(c => {
+ return Control._computeEnergyImpact(
+ c.dispatchesSincePrevious,
+ c.durationSincePrevious
+ );
+ })
+ );
+ },
+};
+
+var View = {
+ _fragment: document.createDocumentFragment(),
+ async commit() {
+ let tbody = document.getElementById("dispatch-tbody");
+
+ // Force translation to happen before we insert the new content in the DOM
+ // to avoid flicker when resizing.
+ await document.l10n.translateFragment(this._fragment);
+
+ while (tbody.firstChild) {
+ tbody.firstChild.remove();
+ }
+ tbody.appendChild(this._fragment);
+ this._fragment = document.createDocumentFragment();
+ },
+ insertAfterRow(row) {
+ row.parentNode.insertBefore(this._fragment, row.nextSibling);
+ this._fragment = document.createDocumentFragment();
+ },
+ displayEnergyImpact(elt, energyImpact, maxEnergyImpact) {
+ if (!energyImpact) {
+ elt.textContent = "–";
+ elt.style.setProperty("--bar-width", 0);
+ } else {
+ let impact;
+ let barWidth;
+ const mediumEnergyImpact = 25;
+ if (energyImpact < 1) {
+ impact = "low";
+ // Width 0-10%.
+ barWidth = 10 * energyImpact;
+ } else if (energyImpact < mediumEnergyImpact) {
+ impact = "medium";
+ // Width 10-50%.
+ barWidth = (10 + 2 * energyImpact) * (5 / 6);
+ } else {
+ impact = "high";
+ // Width 50-100%.
+ let energyImpactFromZero = energyImpact - mediumEnergyImpact;
+ if (maxEnergyImpact > 100) {
+ barWidth =
+ 50 +
+ (energyImpactFromZero / (maxEnergyImpact - mediumEnergyImpact)) *
+ 50;
+ } else {
+ barWidth = 50 + energyImpactFromZero * (2 / 3);
+ }
+ }
+ document.l10n.setAttributes(elt, "energy-impact-" + impact, {
+ value: energyImpact,
+ });
+ if (maxEnergyImpact != -1) {
+ elt.style.setProperty("--bar-width", barWidth);
+ }
+ }
+ },
+ appendRow(
+ name,
+ energyImpact,
+ memory,
+ tooltip,
+ type,
+ maxEnergyImpact = -1,
+ image = ""
+ ) {
+ let row = document.createElement("tr");
+
+ let elt = document.createElement("td");
+ if (typeof name == "string") {
+ elt.textContent = name;
+ } else if (name.title) {
+ document.l10n.setAttributes(elt, name.id, { title: name.title });
+ } else {
+ document.l10n.setAttributes(elt, name.id);
+ }
+ if (image) {
+ elt.style.backgroundImage = `url('${image}')`;
+ }
+
+ if (["subframe", "tracker", "worker"].includes(type)) {
+ elt.classList.add("indent");
+ } else {
+ elt.classList.add("root");
+ }
+ if (["tracker", "worker"].includes(type)) {
+ elt.classList.add(type);
+ }
+ row.appendChild(elt);
+
+ elt = document.createElement("td");
+ let typeLabelType = type == "system-addon" ? "addon" : type;
+ document.l10n.setAttributes(elt, "type-" + typeLabelType);
+ row.appendChild(elt);
+
+ elt = document.createElement("td");
+ elt.classList.add("energy-impact");
+ this.displayEnergyImpact(elt, energyImpact, maxEnergyImpact);
+ row.appendChild(elt);
+
+ elt = document.createElement("td");
+ if (!memory) {
+ elt.textContent = "–";
+ } else {
+ let unit = "KB";
+ memory = Math.ceil(memory / 1024);
+ if (memory > 1024) {
+ memory = Math.ceil((memory / 1024) * 10) / 10;
+ unit = "MB";
+ if (memory > 1024) {
+ memory = Math.ceil((memory / 1024) * 100) / 100;
+ unit = "GB";
+ }
+ }
+ document.l10n.setAttributes(elt, "size-" + unit, { value: memory });
+ }
+ row.appendChild(elt);
+
+ if (tooltip) {
+ for (let key of ["dispatchesSincePrevious", "durationSincePrevious"]) {
+ if (Number.isNaN(tooltip[key]) || tooltip[key] < 0) {
+ tooltip[key] = "–";
+ }
+ }
+ document.l10n.setAttributes(row, "item", tooltip);
+ }
+
+ elt = document.createElement("td");
+ if (type == "tab") {
+ let img = document.createElement("span");
+ img.className = "action-icon close-icon";
+ document.l10n.setAttributes(img, "close-tab");
+ elt.appendChild(img);
+ } else if (type == "addon") {
+ let img = document.createElement("span");
+ img.className = "action-icon addon-icon";
+ document.l10n.setAttributes(img, "show-addon");
+ elt.appendChild(img);
+ }
+ row.appendChild(elt);
+
+ this._fragment.appendChild(row);
+ return row;
+ },
+};
+
+var Control = {
+ _openItems: new Set(),
+ _sortOrder: "",
+ _removeSubtree(row) {
+ while (
+ row.nextSibling &&
+ row.nextSibling.firstChild.classList.contains("indent")
+ ) {
+ row.nextSibling.remove();
+ }
+ },
+ init() {
+ let tbody = document.getElementById("dispatch-tbody");
+ tbody.addEventListener("click", event => {
+ this._updateLastMouseEvent();
+
+ // Handle showing or hiding subitems of a row.
+ let target = event.target;
+ if (target.classList.contains("twisty")) {
+ let row = target.parentNode.parentNode;
+ let id = row.windowId;
+ if (target.classList.toggle("open")) {
+ this._openItems.add(id);
+ this._showChildren(row);
+ View.insertAfterRow(row);
+ } else {
+ this._openItems.delete(id);
+ this._removeSubtree(row);
+ }
+ return;
+ }
+
+ // Handle closing a tab.
+ if (target.classList.contains("close-icon")) {
+ let row = target.parentNode.parentNode;
+ let id = parseInt(row.windowId);
+ let found = tabFinder.get(id);
+ if (!found || !found.tabbrowser) {
+ return;
+ }
+ let { tabbrowser, tab } = found;
+ tabbrowser.removeTab(tab);
+ this._removeSubtree(row);
+ row.remove();
+ return;
+ }
+
+ if (target.classList.contains("addon-icon")) {
+ let row = target.parentNode.parentNode;
+ let id = row.windowId;
+ let parentWin =
+ window.docShell.browsingContext.embedderElement.ownerGlobal;
+ parentWin.BrowserOpenAddonsMgr(
+ "addons://detail/" + encodeURIComponent(id)
+ );
+ return;
+ }
+
+ // Handle selection changes
+ let row = target.parentNode;
+ if (this.selectedRow) {
+ this.selectedRow.removeAttribute("selected");
+ }
+ if (row.windowId) {
+ row.setAttribute("selected", "true");
+ this.selectedRow = row;
+ } else if (this.selectedRow) {
+ this.selectedRow = null;
+ }
+ });
+
+ // Select the tab of double clicked items.
+ tbody.addEventListener("dblclick", event => {
+ let id = parseInt(event.target.parentNode.windowId);
+ if (isNaN(id)) {
+ return;
+ }
+ let found = tabFinder.get(id);
+ if (!found || !found.tabbrowser) {
+ return;
+ }
+ let { tabbrowser, tab } = found;
+ tabbrowser.selectedTab = tab;
+ tabbrowser.ownerGlobal.focus();
+ });
+
+ tbody.addEventListener("mousemove", () => {
+ this._updateLastMouseEvent();
+ });
+
+ window.addEventListener("visibilitychange", event => {
+ if (!document.hidden) {
+ this._updateDisplay(true);
+ }
+ });
+
+ document
+ .getElementById("dispatch-thead")
+ .addEventListener("click", async event => {
+ if (!event.target.classList.contains("clickable")) {
+ return;
+ }
+
+ if (this._sortOrder) {
+ let [column, direction] = this._sortOrder.split("_");
+ const td = document.getElementById(`column-${column}`);
+ td.classList.remove(direction);
+ }
+
+ const columnId = event.target.id;
+ if (columnId == "column-type") {
+ this._sortOrder =
+ this._sortOrder == "type_asc" ? "type_desc" : "type_asc";
+ } else if (columnId == "column-energy-impact") {
+ this._sortOrder =
+ this._sortOrder == "energy-impact_desc"
+ ? "energy-impact_asc"
+ : "energy-impact_desc";
+ } else if (columnId == "column-memory") {
+ this._sortOrder =
+ this._sortOrder == "memory_desc" ? "memory_asc" : "memory_desc";
+ } else if (columnId == "column-name") {
+ this._sortOrder =
+ this._sortOrder == "name_asc" ? "name_desc" : "name_asc";
+ }
+
+ let direction = this._sortOrder.split("_")[1];
+ event.target.classList.add(direction);
+
+ await this._updateDisplay(true);
+ });
+ },
+ _lastMouseEvent: 0,
+ _updateLastMouseEvent() {
+ this._lastMouseEvent = Date.now();
+ },
+ async update() {
+ await State.update();
+
+ if (document.hidden) {
+ return;
+ }
+
+ await wait(0);
+
+ await this._updateDisplay();
+ },
+ // The force parameter can force a full update even when the mouse has been
+ // moved recently.
+ async _updateDisplay(force = false) {
+ let counters = State.getCounters();
+ let maxEnergyImpact = State.getMaxEnergyImpact(counters);
+ // If the mouse has been moved recently, update the data displayed
+ // without moving any item to avoid the risk of users clicking an action
+ // button for the wrong item.
+ // Memory use is unlikely to change dramatically within a few seconds, so
+ // it's probably fine to not update the Memory column in this case.
+ if (
+ !force &&
+ Date.now() - this._lastMouseEvent < TIME_BEFORE_SORTING_AGAIN
+ ) {
+ let energyImpactPerId = new Map();
+ for (let {
+ id,
+ dispatchesSincePrevious,
+ durationSincePrevious,
+ } of counters) {
+ let energyImpact = this._computeEnergyImpact(
+ dispatchesSincePrevious,
+ durationSincePrevious
+ );
+ energyImpactPerId.set(id, energyImpact);
+ }
+
+ let row = document.getElementById("dispatch-tbody").firstChild;
+ while (row) {
+ if (row.windowId && energyImpactPerId.has(row.windowId)) {
+ // We update the value in the Energy Impact column, but don't
+ // update the children, as if the child count changes there's a
+ // risk of making other rows move up or down.
+ const kEnergyImpactColumn = 2;
+ let elt = row.childNodes[kEnergyImpactColumn];
+ View.displayEnergyImpact(
+ elt,
+ energyImpactPerId.get(row.windowId),
+ maxEnergyImpact
+ );
+ }
+ row = row.nextSibling;
+ }
+ return;
+ }
+
+ let selectedId = -1;
+ // Reset the selectedRow field and the _openItems set each time we redraw
+ // to avoid keeping forever references to closed window ids.
+ if (this.selectedRow) {
+ selectedId = this.selectedRow.windowId;
+ this.selectedRow = null;
+ }
+ let openItems = this._openItems;
+ this._openItems = new Set();
+
+ counters = this._sortCounters(counters);
+ for (let {
+ id,
+ name,
+ image,
+ type,
+ totalDispatches,
+ dispatchesSincePrevious,
+ memory,
+ totalDuration,
+ durationSincePrevious,
+ children,
+ } of counters) {
+ let row = View.appendRow(
+ name,
+ this._computeEnergyImpact(
+ dispatchesSincePrevious,
+ durationSincePrevious
+ ),
+ memory,
+ {
+ totalDispatches,
+ totalDuration: Math.ceil(totalDuration / 1000),
+ dispatchesSincePrevious,
+ durationSincePrevious: Math.ceil(durationSincePrevious / 1000),
+ },
+ type,
+ maxEnergyImpact,
+ image
+ );
+ row.windowId = id;
+ if (id == selectedId) {
+ row.setAttribute("selected", "true");
+ this.selectedRow = row;
+ }
+
+ if (!children.length) {
+ continue;
+ }
+
+ // Show the twisty image.
+ let elt = row.firstChild;
+ let img = document.createElement("span");
+ img.className = "twisty";
+ let open = openItems.has(id);
+ if (open) {
+ img.classList.add("open");
+ this._openItems.add(id);
+ }
+
+ // If there's an l10n id on our <td> node, any image we add will be
+ // removed during localization, so move the l10n id to a <span>
+ let l10nAttrs = document.l10n.getAttributes(elt);
+ if (l10nAttrs.id) {
+ let span = document.createElement("span");
+ document.l10n.setAttributes(span, l10nAttrs.id, l10nAttrs.args);
+ elt.removeAttribute("data-l10n-id");
+ elt.removeAttribute("data-l10n-args");
+ elt.insertBefore(span, elt.firstChild);
+ }
+
+ elt.insertBefore(img, elt.firstChild);
+
+ row._children = children;
+ if (open) {
+ this._showChildren(row);
+ }
+ }
+
+ await View.commit();
+ },
+ _showChildren(row) {
+ let children = row._children;
+ children.sort(
+ (a, b) => b.dispatchesSincePrevious - a.dispatchesSincePrevious
+ );
+ for (let row of children) {
+ let host = row.host.replace(/^blob:https?:\/\//, "");
+ let type = "subframe";
+ if (State.isTracker(host)) {
+ type = "tracker";
+ }
+ if (row.isWorker) {
+ type = "worker";
+ }
+ View.appendRow(
+ row.host,
+ this._computeEnergyImpact(
+ row.dispatchesSincePrevious,
+ row.durationSincePrevious
+ ),
+ row.memory,
+ {
+ totalDispatches: row.dispatchCount,
+ totalDuration: Math.ceil(row.duration / 1000),
+ dispatchesSincePrevious: row.dispatchesSincePrevious,
+ durationSincePrevious: Math.ceil(row.durationSincePrevious / 1000),
+ },
+ type
+ );
+ }
+ },
+ _computeEnergyImpact(dispatches, duration) {
+ // 'Dispatches' doesn't make sense to users, and it's difficult to present
+ // two numbers in a meaningful way, so we need to somehow aggregate the
+ // dispatches and duration values we have.
+ // The current formula to aggregate the numbers assumes that the cost of
+ // a dispatch is equivalent to 1ms of CPU time.
+ // Dividing the result by the sampling interval and by 10 gives a number that
+ // looks like a familiar percentage to users, as fullying using one core will
+ // result in a number close to 100.
+ let energyImpact =
+ Math.max(duration || 0, dispatches * 1000) / UPDATE_INTERVAL_MS / 10;
+ // Keep only 2 digits after the decimal point.
+ return Math.ceil(energyImpact * 100) / 100;
+ },
+ _getTypeWeight(type) {
+ let weights = {
+ tab: 3,
+ addon: 2,
+ "system-addon": 1,
+ };
+ return weights[type] || 0;
+ },
+ _sortCounters(counters) {
+ return counters.sort((a, b) => {
+ // Force 'Recently Closed Tabs' to be always at the bottom, because it'll
+ // never be actionable.
+ if (a.name.id && a.name.id == "ghost-windows") {
+ return 1;
+ }
+
+ if (this._sortOrder) {
+ let res;
+ let [column, order] = this._sortOrder.split("_");
+ switch (column) {
+ case "memory":
+ res = a.memory - b.memory;
+ break;
+ case "type":
+ if (a.type != b.type) {
+ res = this._getTypeWeight(b.type) - this._getTypeWeight(a.type);
+ } else {
+ res = String.prototype.localeCompare.call(a.name, b.name);
+ }
+ break;
+ case "name":
+ res = String.prototype.localeCompare.call(a.name, b.name);
+ break;
+ case "energy-impact":
+ res =
+ this._computeEnergyImpact(
+ a.dispatchesSincePrevious,
+ a.durationSincePrevious
+ ) -
+ this._computeEnergyImpact(
+ b.dispatchesSincePrevious,
+ b.durationSincePrevious
+ );
+ break;
+ default:
+ res = String.prototype.localeCompare.call(a.name, b.name);
+ }
+ if (order == "desc") {
+ res = -1 * res;
+ }
+ return res;
+ }
+
+ // Note: _computeEnergyImpact uses UPDATE_INTERVAL_MS which doesn't match
+ // the time between the most recent sample and the start of the buffer,
+ // BUFFER_DURATION_MS would be better, but the values is never displayed
+ // so this is OK.
+ let aEI = this._computeEnergyImpact(
+ a.dispatchesSinceStartOfBuffer,
+ a.durationSinceStartOfBuffer
+ );
+ let bEI = this._computeEnergyImpact(
+ b.dispatchesSinceStartOfBuffer,
+ b.durationSinceStartOfBuffer
+ );
+ if (aEI != bEI) {
+ return bEI - aEI;
+ }
+
+ // a.name is sometimes an object, so we can't use a.name.localeCompare.
+ return String.prototype.localeCompare.call(a.name, b.name);
+ });
+ },
+};
+
+window.onload = async function() {
+ Control.init();
+
+ let addons = await AddonManager.getAddonsByTypes(["extension"]);
+ for (let addon of addons) {
+ if (addon.isSystem) {
+ gSystemAddonIds.add(addon.id);
+ }
+ }
+
+ await Control.update();
+ window.setInterval(() => Control.update(), UPDATE_INTERVAL_MS);
+};
diff --git a/toolkit/components/aboutperformance/jar.mn b/toolkit/components/aboutperformance/jar.mn
new file mode 100644
index 0000000000..6812a70593
--- /dev/null
+++ b/toolkit/components/aboutperformance/jar.mn
@@ -0,0 +1,8 @@
+# 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/.
+
+toolkit.jar:
+ content/global/aboutPerformance.html (content/aboutPerformance.html)
+ content/global/aboutPerformance.js (content/aboutPerformance.js)
+ content/global/aboutPerformance.css (content/aboutPerformance.css)
diff --git a/toolkit/components/aboutperformance/moz.build b/toolkit/components/aboutperformance/moz.build
new file mode 100644
index 0000000000..f397ead22d
--- /dev/null
+++ b/toolkit/components/aboutperformance/moz.build
@@ -0,0 +1,12 @@
+# -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
+# vim: set filetype=python:
+# 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/.
+
+with Files("**"):
+ BUG_COMPONENT = ("Toolkit", "Performance Monitoring")
+
+JAR_MANIFESTS += ["jar.mn"]
+
+BROWSER_CHROME_MANIFESTS += ["tests/browser/browser.ini"]
diff --git a/toolkit/components/aboutperformance/tests/browser/browser.ini b/toolkit/components/aboutperformance/tests/browser/browser.ini
new file mode 100644
index 0000000000..c36144fc2a
--- /dev/null
+++ b/toolkit/components/aboutperformance/tests/browser/browser.ini
@@ -0,0 +1,7 @@
+[DEFAULT]
+support-files =
+ browser_compartments.html
+ browser_compartments_frame.html
+ browser_compartments_script.js
+
+[browser_aboutperformance.js]
diff --git a/toolkit/components/aboutperformance/tests/browser/browser_aboutperformance.js b/toolkit/components/aboutperformance/tests/browser/browser_aboutperformance.js
new file mode 100644
index 0000000000..ae4e1d0bae
--- /dev/null
+++ b/toolkit/components/aboutperformance/tests/browser/browser_aboutperformance.js
@@ -0,0 +1,106 @@
+/* Any copyright is dedicated to the Public Domain.
+ * http://creativecommons.org/publicdomain/zero/1.0/ */
+
+"use strict";
+
+const URL =
+ "http://example.com/browser/toolkit/components/aboutperformance/tests/browser/browser_compartments.html?test=" +
+ Math.random();
+
+add_task(async function init() {
+ info("Setting up about:performance");
+ let tabAboutPerformance = (gBrowser.selectedTab = BrowserTestUtils.addTab(
+ gBrowser,
+ "about:performance"
+ ));
+
+ await BrowserTestUtils.browserLoaded(tabAboutPerformance.linkedBrowser);
+
+ info(`Setting up ${URL}`);
+ let tabContent = BrowserTestUtils.addTab(gBrowser, URL);
+ await BrowserTestUtils.browserLoaded(tabContent.linkedBrowser);
+
+ let doc = tabAboutPerformance.linkedBrowser.contentDocument;
+ let tbody = doc.getElementById("dispatch-tbody");
+
+ // Wait until the table has first been populated.
+ await TestUtils.waitForCondition(() => tbody.childElementCount);
+
+ // And wait for another update using a mutation observer, to give our newly created test tab some time
+ // to burn some CPU.
+ await new Promise(resolve => {
+ let observer = new doc.ownerGlobal.MutationObserver(() => {
+ observer.disconnect();
+ resolve();
+ });
+ observer.observe(tbody, { childList: true });
+ });
+
+ // Find the row for our test tab.
+ let row = tbody.firstChild;
+ while (
+ row &&
+ row.firstChild.textContent !=
+ "Main frame for test browser_aboutperformance.js"
+ ) {
+ row = row.nextSibling;
+ }
+
+ Assert.ok(row, "found a table row for our test tab");
+ Assert.equal(
+ row.windowId,
+ tabContent.linkedBrowser.outerWindowID,
+ "the correct window id is set"
+ );
+
+ // Ensure it is reported as a medium or high energy impact.
+ let l10nId = row.children[2].getAttribute("data-l10n-id");
+ Assert.ok(
+ ["energy-impact-medium", "energy-impact-high"].includes(l10nId),
+ "our test tab is medium or high energy impact"
+ );
+
+ // Verify selecting a row works.
+ EventUtils.synthesizeMouseAtCenter(
+ row,
+ {},
+ tabAboutPerformance.linkedBrowser.contentWindow
+ );
+
+ Assert.equal(
+ row.getAttribute("selected"),
+ "true",
+ "doing a single click selects the row"
+ );
+
+ // Verify selecting a tab with a double click.
+ Assert.equal(
+ gBrowser.selectedTab,
+ tabAboutPerformance,
+ "the about:performance tab is selected"
+ );
+ EventUtils.synthesizeMouseAtCenter(
+ row,
+ { clickCount: 2 },
+ tabAboutPerformance.linkedBrowser.contentWindow
+ );
+ Assert.equal(
+ gBrowser.selectedTab,
+ tabContent,
+ "after a double click the test tab is selected"
+ );
+
+ // Verify we can close a tab using the X button.
+ // Switch back to about:performance...
+ await BrowserTestUtils.switchTab(gBrowser, tabAboutPerformance);
+ // ... and click the X button at the end of the row.
+ let tabClosing = BrowserTestUtils.waitForTabClosing(tabContent);
+ EventUtils.synthesizeMouseAtCenter(
+ row.children[4],
+ {},
+ tabAboutPerformance.linkedBrowser.contentWindow
+ );
+ await tabClosing;
+
+ BrowserTestUtils.removeTab(tabAboutPerformance);
+});
diff --git a/toolkit/components/aboutperformance/tests/browser/browser_compartments.html b/toolkit/components/aboutperformance/tests/browser/browser_compartments.html
new file mode 100644
index 0000000000..379422d7ac
--- /dev/null
+++ b/toolkit/components/aboutperformance/tests/browser/browser_compartments.html
@@ -0,0 +1,21 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <meta charset="UTF-8">
+ <title>
+ Main frame for test browser_aboutperformance.js
+ </title>
+</head>
+<body>
+Main frame.
+
+<iframe src="browser_compartments_frame.html?frame=1">
+ Subframe 1
+</iframe>
+
+<iframe src="browser_compartments_frame.html?frame=2">
+ Subframe 2.
+</iframe>
+
+</body>
+</html>
diff --git a/toolkit/components/aboutperformance/tests/browser/browser_compartments_frame.html b/toolkit/components/aboutperformance/tests/browser/browser_compartments_frame.html
new file mode 100644
index 0000000000..44a073d3bb
--- /dev/null
+++ b/toolkit/components/aboutperformance/tests/browser/browser_compartments_frame.html
@@ -0,0 +1,13 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <meta charset="UTF-8">
+ <title>
+ Subframe for test browser_compartments.html (do not change this title)
+ </title>
+ <script src="browser_compartments_script.js"></script>
+</head>
+<body>
+Subframe loaded.
+</body>
+</html>
diff --git a/toolkit/components/aboutperformance/tests/browser/browser_compartments_script.js b/toolkit/components/aboutperformance/tests/browser/browser_compartments_script.js
new file mode 100644
index 0000000000..2547a1a010
--- /dev/null
+++ b/toolkit/components/aboutperformance/tests/browser/browser_compartments_script.js
@@ -0,0 +1,11 @@
+// Use some CPU.
+var interval = window.setInterval(() => {
+ // Compute an arbitrary value, print it out to make sure that the JS
+ // engine doesn't discard all our computation.
+ var date = Date.now();
+ var array = [];
+ var i = 0;
+ while (Date.now() - date <= 100) {
+ array[i % 2] = i++;
+ }
+}, 300);