fune/toolkit/components/backgroundhangmonitor/tests/test_BHRObserver.js
2019-09-14 09:39:26 +00:00

168 lines
5.4 KiB
JavaScript

/* Any copyright is dedicated to the Public Domain.
http://creativecommons.org/publicdomain/zero/1.0/ */
const { Services } = ChromeUtils.import("resource://gre/modules/Services.jsm");
const { TelemetryUtils } = ChromeUtils.import(
"resource://gre/modules/TelemetryUtils.jsm"
);
function ensureProfilerInitialized() {
// Starting and stopping the profiler with the "stackwalk" flag will cause the
// profiler's stackwalking features to be synchronously initialized. This
// should prevent us from not initializing BHR quickly enough.
if (!Services.profiler.CanProfile()) {
return false;
}
let features = ["stackwalk"];
Services.profiler.StartProfiler(1000, 10, features);
Services.profiler.StopProfiler();
return true;
}
add_task(async function test_BHRObserver() {
if (!Services.telemetry.canRecordExtended) {
ok("Hang reporting not enabled.");
return;
}
if (!ensureProfilerInitialized()) {
return;
}
let telSvc = Cc["@mozilla.org/bhr-telemetry-service;1"].getService()
.wrappedJSObject;
ok(telSvc, "Should have BHRTelemetryService");
let beforeLen = telSvc.payload.hangs.length;
if (Services.appinfo.OS === "Linux" || Services.appinfo.OS === "Android") {
// We use the rt_tgsigqueueinfo syscall on Linux which requires a
// certain kernel version. It's not an error if the system running
// the test is older than that.
let kernel =
Services.sysinfo.get("kernel_version") || Services.sysinfo.get("version");
if (Services.vc.compare(kernel, "2.6.31") < 0) {
ok("Hang reporting not supported for old kernel.");
return;
}
}
let hangsPromise = new Promise(resolve => {
let hangs = [];
const onThreadHang = subject => {
let hang = subject.QueryInterface(Ci.nsIHangDetails);
if (hang.thread.startsWith("Gecko")) {
hangs.push(hang);
if (hangs.length >= 3) {
Services.obs.removeObserver(onThreadHang, "bhr-thread-hang");
resolve(hangs);
}
}
};
Services.obs.addObserver(onThreadHang, "bhr-thread-hang");
});
// We're going to trigger two hangs, of various lengths. One should be a
// transient hang, and the other a permanent hang. We'll wait for the hangs to
// be recorded.
executeSoon(() => {
let startTime = Date.now();
// eslint-disable-next-line no-empty
while (Date.now() - startTime < 10000) {}
});
executeSoon(() => {
let startTime = Date.now();
// eslint-disable-next-line no-empty
while (Date.now() - startTime < 1000) {}
});
Services.prefs.setBoolPref(
TelemetryUtils.Preferences.OverridePreRelease,
true
);
let childDone = run_test_in_child("child_cause_hang.js");
// Now we wait for the hangs to have their bhr-thread-hang message fired for
// them, collect them, and analyize the response.
let hangs = await hangsPromise;
equal(hangs.length, 3);
hangs.forEach(hang => {
ok(hang.duration > 0);
ok(hang.thread == "Gecko" || hang.thread == "Gecko_Child");
equal(typeof hang.runnableName, "string");
// hang.stack
ok(Array.isArray(hang.stack));
ok(!!hang.stack.length);
hang.stack.forEach(entry => {
// Each stack frame entry is either a native or pseudostack entry. A
// native stack entry is an array with module index (number), and offset
// (hex string), while the pseudostack entry is a bare string.
if (Array.isArray(entry)) {
equal(entry.length, 2);
equal(typeof entry[0], "number");
equal(typeof entry[1], "string");
} else {
equal(typeof entry, "string");
}
});
// hang.modules
ok(Array.isArray(hang.modules));
hang.modules.forEach(module => {
ok(Array.isArray(module));
equal(module.length, 2);
equal(typeof module[0], "string");
equal(typeof module[1], "string");
});
// hang.annotations
equal(typeof hang.annotations, "object");
Object.keys(hang.annotations).forEach(key => {
equal(typeof hang.annotations[key], "string");
});
});
// Check that the telemetry service collected pings which make sense
ok(telSvc.payload.hangs.length - beforeLen >= 3);
ok(Array.isArray(telSvc.payload.modules));
telSvc.payload.modules.forEach(module => {
ok(Array.isArray(module));
equal(module.length, 2);
equal(typeof module[0], "string");
equal(typeof module[1], "string");
});
telSvc.payload.hangs.forEach(hang => {
ok(hang.duration > 0);
ok(hang.thread == "Gecko" || hang.thread == "Gecko_Child");
equal(typeof hang.runnableName, "string");
// hang.stack
ok(Array.isArray(hang.stack));
ok(!!hang.stack.length);
hang.stack.forEach(entry => {
// Each stack frame entry is either a native or pseudostack entry. A
// native stack entry is an array with module index (number), and offset
// (hex string), while the pseudostack entry is a bare string.
if (Array.isArray(entry)) {
equal(entry.length, 2);
equal(typeof entry[0], "number");
ok(entry[0] < telSvc.payload.hangs.length);
equal(typeof entry[1], "string");
} else {
equal(typeof entry, "string");
}
});
// hang.annotations
equal(typeof hang.annotations, "object");
Object.keys(hang.annotations).forEach(key => {
equal(typeof hang.annotations[key], "string");
});
});
do_send_remote_message("bhr_hangs_detected");
await childDone;
});