summaryrefslogtreecommitdiffstats
path: root/toolkit/components/backgroundhangmonitor/BHRTelemetryService.jsm
diff options
context:
space:
mode:
Diffstat (limited to 'toolkit/components/backgroundhangmonitor/BHRTelemetryService.jsm')
-rw-r--r--toolkit/components/backgroundhangmonitor/BHRTelemetryService.jsm168
1 files changed, 168 insertions, 0 deletions
diff --git a/toolkit/components/backgroundhangmonitor/BHRTelemetryService.jsm b/toolkit/components/backgroundhangmonitor/BHRTelemetryService.jsm
new file mode 100644
index 0000000000..58e5692699
--- /dev/null
+++ b/toolkit/components/backgroundhangmonitor/BHRTelemetryService.jsm
@@ -0,0 +1,168 @@
+/* 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";
+
+ChromeUtils.import("resource://gre/modules/Services.jsm", this);
+
+ChromeUtils.defineModuleGetter(this, "OS", "resource://gre/modules/osfile.jsm");
+ChromeUtils.defineModuleGetter(
+ this,
+ "TelemetryController",
+ "resource://gre/modules/TelemetryController.jsm"
+);
+
+function BHRTelemetryService() {
+ // Allow tests to get access to this object to verify it works correctly.
+ this.wrappedJSObject = this;
+
+ Services.obs.addObserver(this, "profile-before-change");
+ Services.obs.addObserver(this, "bhr-thread-hang");
+ Services.obs.addObserver(this, "idle-daily");
+
+ this.resetPayload();
+}
+
+BHRTelemetryService.prototype = Object.freeze({
+ classID: Components.ID("{117c8cdf-69e6-4f31-a439-b8a654c67127}"),
+ QueryInterface: ChromeUtils.generateQI(["nsIObserver"]),
+
+ TRANSMIT_HANG_COUNT: 50,
+
+ resetPayload() {
+ this.startTime = +new Date();
+ this.payload = {
+ modules: [],
+ hangs: [],
+ };
+ this.clearPermahangFile = false;
+ },
+
+ recordHang({
+ duration,
+ thread,
+ runnableName,
+ process,
+ stack,
+ remoteType,
+ modules,
+ annotations,
+ wasPersisted,
+ }) {
+ if (!Services.telemetry.canRecordExtended) {
+ return;
+ }
+
+ // Create a mapping from module indicies in the original nsIHangDetails
+ // object to this.payload.modules indicies.
+ let moduleIdxs = modules.map(module => {
+ let idx = this.payload.modules.findIndex(m => {
+ return m[0] === module[0] && m[1] === module[1];
+ });
+ if (idx === -1) {
+ idx = this.payload.modules.length;
+ this.payload.modules.push(module);
+ }
+ return idx;
+ });
+
+ // Native stack frames are [modIdx, offset] arrays. If we have a valid
+ // module index, we want to map it to the this.payload.modules array.
+ for (let i = 0; i < stack.length; ++i) {
+ if (Array.isArray(stack[i]) && stack[i][0] !== -1) {
+ stack[i][0] = moduleIdxs[stack[i][0]];
+ } else if (typeof stack[i] == "string") {
+ // This is just a precaution - we don't currently know of sensitive
+ // URLs being included in label frames' dynamic strings which we
+ // include here, but this is just an added guard. Here we strip any
+ // string with a :// in it that isn't a chrome:// or resource://
+ // URL. This is not completely robust, but we are already trying to
+ // protect against this by only including dynamic strings from the
+ // opt-in AUTO_PROFILER_..._NONSENSITIVE macros.
+ let match = /[^\s]+:\/\/.*/.exec(stack[i]);
+ if (
+ match &&
+ !match[0].startsWith("chrome://") &&
+ !match[0].startsWith("resource://")
+ ) {
+ stack[i] = stack[i].replace(match[0], "(excluded)");
+ }
+ }
+ }
+
+ // Create the hang object to record in the payload.
+ this.payload.hangs.push({
+ duration,
+ thread,
+ runnableName,
+ process,
+ remoteType,
+ annotations,
+ stack,
+ });
+
+ if (wasPersisted) {
+ this.clearPermahangFile = true;
+ }
+
+ // If we have collected enough hangs, we can submit the hangs we have
+ // collected to telemetry.
+ if (this.payload.hangs.length > this.TRANSMIT_HANG_COUNT) {
+ this.submit();
+ }
+ },
+
+ submit() {
+ if (this.clearPermahangFile) {
+ OS.File.remove(
+ OS.Path.join(OS.Constants.Path.profileDir, "last_permahang.bin"),
+ { ignoreAbsent: true }
+ );
+ }
+
+ if (!Services.telemetry.canRecordExtended) {
+ return;
+ }
+
+ // NOTE: We check a separate bhrPing.enabled pref here. This pref is unset
+ // when running tests so that we run as much of BHR as possible (to catch
+ // errors) while avoiding timeouts caused by invoking `pingsender` during
+ // testing.
+ if (
+ Services.prefs.getBoolPref("toolkit.telemetry.bhrPing.enabled", false)
+ ) {
+ this.payload.timeSinceLastPing = new Date() - this.startTime;
+ TelemetryController.submitExternalPing("bhr", this.payload, {
+ addEnvironment: true,
+ });
+ }
+ this.resetPayload();
+ },
+
+ shutdown() {
+ Services.obs.removeObserver(this, "profile-before-change");
+ Services.obs.removeObserver(this, "bhr-thread-hang");
+ Services.obs.removeObserver(this, "idle-daily");
+ this.submit();
+ },
+
+ observe(aSubject, aTopic, aData) {
+ switch (aTopic) {
+ case "profile-after-change":
+ this.resetPayload();
+ break;
+ case "bhr-thread-hang":
+ this.recordHang(aSubject.QueryInterface(Ci.nsIHangDetails));
+ break;
+ case "profile-before-change":
+ this.shutdown();
+ break;
+ case "idle-daily":
+ this.submit();
+ break;
+ }
+ },
+});
+
+var EXPORTED_SYMBOLS = ["BHRTelemetryService"];