summaryrefslogtreecommitdiffstats
path: root/layout/tools/reftest/manifest.sys.mjs
diff options
context:
space:
mode:
Diffstat (limited to 'layout/tools/reftest/manifest.sys.mjs')
-rw-r--r--layout/tools/reftest/manifest.sys.mjs1046
1 files changed, 1046 insertions, 0 deletions
diff --git a/layout/tools/reftest/manifest.sys.mjs b/layout/tools/reftest/manifest.sys.mjs
new file mode 100644
index 0000000000..4be0afde57
--- /dev/null
+++ b/layout/tools/reftest/manifest.sys.mjs
@@ -0,0 +1,1046 @@
+/* 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 { globals } from "resource://reftest/globals.sys.mjs";
+
+const {
+ NS_GFXINFO_CONTRACTID,
+
+ TYPE_REFTEST_EQUAL,
+ TYPE_REFTEST_NOTEQUAL,
+ TYPE_LOAD,
+ TYPE_SCRIPT,
+ TYPE_PRINT,
+
+ EXPECTED_PASS,
+ EXPECTED_FAIL,
+ EXPECTED_RANDOM,
+ EXPECTED_FUZZY,
+
+ PREF_BOOLEAN,
+ PREF_STRING,
+ PREF_INTEGER,
+
+ FOCUS_FILTER_NEEDS_FOCUS_TESTS,
+ FOCUS_FILTER_NON_NEEDS_FOCUS_TESTS,
+
+ g,
+} = globals;
+
+import { NetUtil } from "resource://gre/modules/NetUtil.sys.mjs";
+import { AppConstants } from "resource://gre/modules/AppConstants.sys.mjs";
+
+const NS_NETWORK_PROTOCOL_CONTRACTID_PREFIX =
+ "@mozilla.org/network/protocol;1?name=";
+
+const RE_PROTOCOL = /^\w+:/;
+const RE_PREF_ITEM = /^(|test-|ref-)pref\((.+?),(.*)\)$/;
+
+export function ReadTopManifest(aFileURL, aFilter, aManifestID) {
+ var url = g.ioService.newURI(aFileURL);
+ if (!url) {
+ throw new Error("Expected a file or http URL for the manifest.");
+ }
+
+ g.manifestsLoaded = {};
+ ReadManifest(url, aFilter, aManifestID);
+}
+
+// Note: If you materially change the reftest manifest parsing,
+// please keep the parser in layout/tools/reftest/__init__.py in sync.
+// eslint-disable-next-line complexity
+function ReadManifest(aURL, aFilter, aManifestID) {
+ // Ensure each manifest is only read once. This assumes that manifests that
+ // are included with filters will be read via their include before they are
+ // read directly in the case of a duplicate
+ if (g.manifestsLoaded.hasOwnProperty(aURL.spec)) {
+ if (g.manifestsLoaded[aURL.spec] === null) {
+ return;
+ }
+ aFilter = [aFilter[0], aFilter[1], true];
+ }
+ g.manifestsLoaded[aURL.spec] = aFilter[1];
+
+ var listURL = aURL;
+ var channel = NetUtil.newChannel({
+ uri: aURL,
+ loadUsingSystemPrincipal: true,
+ });
+ try {
+ var inputStream = channel.open();
+ } catch (e) {
+ g.logger.error("failed to open manifest at : " + aURL.spec);
+ throw new Error(e);
+ }
+ if (channel instanceof Ci.nsIHttpChannel && channel.responseStatus != 200) {
+ g.logger.error("HTTP ERROR : " + channel.responseStatus);
+ }
+ var streamBuf = getStreamContent(inputStream);
+ inputStream.close();
+ var lines = streamBuf.split(/\n|\r|\r\n/);
+
+ // The sandbox for fails-if(), etc., condition evaluation. This is not
+ // always required and so is created on demand.
+ var sandbox;
+ function GetOrCreateSandbox() {
+ if (!sandbox) {
+ sandbox = BuildConditionSandbox(aURL);
+ }
+ return sandbox;
+ }
+
+ var lineNo = 0;
+ var urlprefix = "";
+ var defaults = [];
+ var defaultTestPrefSettings = [],
+ defaultRefPrefSettings = [];
+ if (g.compareRetainedDisplayLists) {
+ AddRetainedDisplayListTestPrefs(
+ GetOrCreateSandbox(),
+ defaultTestPrefSettings,
+ defaultRefPrefSettings
+ );
+ }
+ for (var str of lines) {
+ ++lineNo;
+ if (str.charAt(0) == "#") {
+ continue;
+ } // entire line was a comment
+ var i = str.search(/\s+#/);
+ if (i >= 0) {
+ str = str.substring(0, i);
+ }
+ // strip leading and trailing whitespace
+ str = str.replace(/^\s*/, "").replace(/\s*$/, "");
+ if (!str || str == "") {
+ continue;
+ }
+ var items = str.split(/\s+/); // split on whitespace
+
+ if (items[0] == "url-prefix") {
+ if (items.length != 2) {
+ throw new Error(
+ "url-prefix requires one url in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo
+ );
+ }
+ urlprefix = items[1];
+ continue;
+ }
+
+ if (items[0] == "defaults") {
+ items.shift();
+ defaults = items;
+ continue;
+ }
+
+ var expected_status = EXPECTED_PASS;
+ var allow_silent_fail = false;
+ var minAsserts = 0;
+ var maxAsserts = 0;
+ var needs_focus = false;
+ var slow = false;
+ var skip = false;
+ var testPrefSettings = defaultTestPrefSettings.concat();
+ var refPrefSettings = defaultRefPrefSettings.concat();
+ var fuzzy_delta = { min: 0, max: 2 };
+ var fuzzy_pixels = { min: 0, max: 1 };
+ var chaosMode = false;
+ var wrCapture = { test: false, ref: false };
+ var nonSkipUsed = false;
+ var noAutoFuzz = false;
+
+ var origLength = items.length;
+ items = defaults.concat(items);
+ while (
+ items[0].match(
+ /^(fails|needs-focus|random|skip|asserts|slow|require-or|silentfail|pref|test-pref|ref-pref|fuzzy|chaos-mode|wr-capture|wr-capture-ref|noautofuzz)/
+ )
+ ) {
+ var item = items.shift();
+ var stat;
+ var cond;
+ var m = item.match(/^(fails|random|skip|silentfail)-if(\(.*\))$/);
+ if (m) {
+ stat = m[1];
+ // Note: m[2] contains the parentheses, and we want them.
+ cond = Cu.evalInSandbox(m[2], GetOrCreateSandbox());
+ } else if (item.match(/^(fails|random|skip)$/)) {
+ stat = item;
+ cond = true;
+ } else if (item == "needs-focus") {
+ needs_focus = true;
+ cond = false;
+ } else if ((m = item.match(/^asserts\((\d+)(-\d+)?\)$/))) {
+ cond = false;
+ minAsserts = Number(m[1]);
+ maxAsserts = m[2] == undefined ? minAsserts : Number(m[2].substring(1));
+ } else if ((m = item.match(/^asserts-if\((.*?),(\d+)(-\d+)?\)$/))) {
+ cond = false;
+ if (Cu.evalInSandbox("(" + m[1] + ")", GetOrCreateSandbox())) {
+ minAsserts = Number(m[2]);
+ maxAsserts =
+ m[3] == undefined ? minAsserts : Number(m[3].substring(1));
+ }
+ } else if (item == "slow") {
+ cond = false;
+ slow = true;
+ } else if ((m = item.match(/^require-or\((.*?)\)$/))) {
+ var args = m[1].split(/,/);
+ if (args.length != 2) {
+ throw new Error(
+ "Error in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ": wrong number of args to require-or"
+ );
+ }
+ var [precondition_str, fallback_action] = args;
+ var preconditions = precondition_str.split(/&&/);
+ cond = false;
+ for (var precondition of preconditions) {
+ if (precondition === "debugMode") {
+ // Currently unimplemented. Requires asynchronous
+ // JSD call + getting an event while no JS is running
+ stat = fallback_action;
+ cond = true;
+ break;
+ } else if (precondition === "true") {
+ // For testing
+ } else {
+ // Unknown precondition. Assume it is unimplemented.
+ stat = fallback_action;
+ cond = true;
+ break;
+ }
+ }
+ } else if ((m = item.match(/^slow-if\((.*?)\)$/))) {
+ cond = false;
+ if (Cu.evalInSandbox("(" + m[1] + ")", GetOrCreateSandbox())) {
+ slow = true;
+ }
+ } else if (item == "silentfail") {
+ cond = false;
+ allow_silent_fail = true;
+ } else if ((m = item.match(RE_PREF_ITEM))) {
+ cond = false;
+ if (
+ !AddPrefSettings(
+ m[1],
+ m[2],
+ m[3],
+ GetOrCreateSandbox(),
+ testPrefSettings,
+ refPrefSettings
+ )
+ ) {
+ throw new Error(
+ "Error in pref value in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo
+ );
+ }
+ } else if ((m = item.match(/^fuzzy\((\d+)-(\d+),(\d+)-(\d+)\)$/))) {
+ cond = false;
+ expected_status = EXPECTED_FUZZY;
+ fuzzy_delta = ExtractRange(m, 1);
+ fuzzy_pixels = ExtractRange(m, 3);
+ } else if (
+ (m = item.match(/^fuzzy-if\((.*?),(\d+)-(\d+),(\d+)-(\d+)\)$/))
+ ) {
+ cond = false;
+ if (Cu.evalInSandbox("(" + m[1] + ")", GetOrCreateSandbox())) {
+ expected_status = EXPECTED_FUZZY;
+ fuzzy_delta = ExtractRange(m, 2);
+ fuzzy_pixels = ExtractRange(m, 4);
+ }
+ } else if (item == "chaos-mode") {
+ cond = false;
+ chaosMode = true;
+ } else if (item == "wr-capture") {
+ cond = false;
+ wrCapture.test = true;
+ } else if (item == "wr-capture-ref") {
+ cond = false;
+ wrCapture.ref = true;
+ } else if (item == "noautofuzz") {
+ cond = false;
+ noAutoFuzz = true;
+ } else {
+ throw new Error(
+ "Error in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ": unexpected item " +
+ item
+ );
+ }
+
+ if (stat != "skip") {
+ nonSkipUsed = true;
+ }
+
+ if (cond) {
+ if (stat == "fails") {
+ expected_status = EXPECTED_FAIL;
+ } else if (stat == "random") {
+ expected_status = EXPECTED_RANDOM;
+ } else if (stat == "skip") {
+ skip = true;
+ } else if (stat == "silentfail") {
+ allow_silent_fail = true;
+ }
+ }
+ }
+
+ if (items.length > origLength) {
+ // Implies we broke out of the loop before we finished processing
+ // defaults. This means defaults contained an invalid token.
+ throw new Error(
+ "Error in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ": invalid defaults token '" +
+ items[0] +
+ "'"
+ );
+ }
+
+ if (minAsserts > maxAsserts) {
+ throw new Error(
+ "Bad range in manifest file " + aURL.spec + " line " + lineNo
+ );
+ }
+
+ var runHttp = false;
+ var httpDepth;
+ if (items[0] == "HTTP") {
+ runHttp = aURL.scheme == "file"; // We can't yet run the local HTTP server
+ // for non-local reftests.
+ httpDepth = 0;
+ items.shift();
+ } else if (items[0].match(/HTTP\(\.\.(\/\.\.)*\)/)) {
+ // Accept HTTP(..), HTTP(../..), HTTP(../../..), etc.
+ runHttp = aURL.scheme == "file"; // We can't yet run the local HTTP server
+ // for non-local reftests.
+ httpDepth = (items[0].length - 5) / 3;
+ items.shift();
+ }
+
+ // do not prefix the url for include commands or urls specifying
+ // a protocol
+ if (urlprefix && items[0] != "include") {
+ if (items.length > 1 && !items[1].match(RE_PROTOCOL)) {
+ items[1] = urlprefix + items[1];
+ }
+ if (items.length > 2 && !items[2].match(RE_PROTOCOL)) {
+ items[2] = urlprefix + items[2];
+ }
+ }
+
+ var principal = Services.scriptSecurityManager.createContentPrincipal(
+ aURL,
+ {}
+ );
+
+ if (items[0] == "include") {
+ if (items.length != 2) {
+ throw new Error(
+ "Error in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ": incorrect number of arguments to include"
+ );
+ }
+ if (runHttp) {
+ throw new Error(
+ "Error in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ": use of include with http"
+ );
+ }
+
+ // If the expected_status is EXPECTED_PASS (the default) then allow
+ // the include. If 'skip' is true, that means there was a skip
+ // or skip-if annotation (with a true condition) on this include
+ // statement, so we should skip the include. Any other expected_status
+ // is disallowed since it's nonintuitive as to what the intended
+ // effect is.
+ if (nonSkipUsed) {
+ throw new Error(
+ "Error in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ": include statement with annotation other than 'skip' or 'skip-if'"
+ );
+ } else if (skip) {
+ g.logger.info(
+ "Skipping included manifest at " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ " due to matching skip condition"
+ );
+ } else {
+ // poor man's assertion
+ if (expected_status != EXPECTED_PASS) {
+ throw new Error(
+ "Error in manifest file parsing code: we should never get expected_status=" +
+ expected_status +
+ " when nonSkipUsed=false (from " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ")"
+ );
+ }
+
+ var incURI = g.ioService.newURI(items[1], null, listURL);
+ Services.scriptSecurityManager.checkLoadURIWithPrincipal(
+ principal,
+ incURI,
+ Ci.nsIScriptSecurityManager.DISALLOW_SCRIPT
+ );
+
+ // Cannot use nsIFile or similar to manipulate the manifest ID; although it appears
+ // path-like, it does not refer to an actual path in the filesystem.
+ var newManifestID = aManifestID;
+ var included = items[1];
+ // Remove included manifest file name.
+ // eg. dir1/dir2/reftest.list -> dir1/dir2
+ var pos = included.lastIndexOf("/");
+ if (pos <= 0) {
+ included = "";
+ } else {
+ included = included.substring(0, pos);
+ }
+ // Simplify references to parent directories.
+ // eg. dir1/dir2/../dir3 -> dir1/dir3
+ while (included.startsWith("../")) {
+ pos = newManifestID.lastIndexOf("/");
+ if (pos < 0) {
+ pos = 0;
+ }
+ newManifestID = newManifestID.substring(0, pos);
+ included = included.substring(3);
+ }
+ // Use a new manifest ID if the included manifest is in a different directory.
+ if (included.length) {
+ if (newManifestID.length) {
+ newManifestID = newManifestID + "/" + included;
+ } else {
+ // parent directory includes may refer to the topsrcdir
+ newManifestID = included;
+ }
+ }
+ ReadManifest(incURI, aFilter, newManifestID);
+ }
+ } else if (items[0] == TYPE_LOAD || items[0] == TYPE_SCRIPT) {
+ let type = items[0];
+ if (items.length != 2) {
+ throw new Error(
+ "Error in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ": incorrect number of arguments to " +
+ type
+ );
+ }
+ if (type == TYPE_LOAD && expected_status != EXPECTED_PASS) {
+ throw new Error(
+ "Error in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ": incorrect known failure type for load test"
+ );
+ }
+ AddTestItem(
+ {
+ type,
+ expected: expected_status,
+ manifest: aURL.spec,
+ manifestID: TestIdentifier(aURL.spec, aManifestID),
+ allowSilentFail: allow_silent_fail,
+ minAsserts,
+ maxAsserts,
+ needsFocus: needs_focus,
+ slow,
+ skip,
+ prefSettings1: testPrefSettings,
+ prefSettings2: refPrefSettings,
+ fuzzyMinDelta: fuzzy_delta.min,
+ fuzzyMaxDelta: fuzzy_delta.max,
+ fuzzyMinPixels: fuzzy_pixels.min,
+ fuzzyMaxPixels: fuzzy_pixels.max,
+ runHttp,
+ httpDepth,
+ url1: items[1],
+ url2: null,
+ chaosMode,
+ wrCapture,
+ noAutoFuzz,
+ },
+ aFilter,
+ aManifestID
+ );
+ } else if (
+ items[0] == TYPE_REFTEST_EQUAL ||
+ items[0] == TYPE_REFTEST_NOTEQUAL ||
+ items[0] == TYPE_PRINT
+ ) {
+ if (items.length != 3) {
+ throw new Error(
+ "Error in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ": incorrect number of arguments to " +
+ items[0]
+ );
+ }
+
+ if (
+ items[0] == TYPE_REFTEST_NOTEQUAL &&
+ expected_status == EXPECTED_FUZZY &&
+ (fuzzy_delta.min > 0 || fuzzy_pixels.min > 0)
+ ) {
+ throw new Error(
+ "Error in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ": minimum fuzz must be zero for tests of type " +
+ items[0]
+ );
+ }
+
+ let type = items[0];
+ if (g.compareRetainedDisplayLists) {
+ type = TYPE_REFTEST_EQUAL;
+
+ // We expect twice as many assertion failures when comparing
+ // tests because we run each test twice.
+ minAsserts *= 2;
+ maxAsserts *= 2;
+
+ // Skip the test if it is expected to fail in both modes.
+ // It would unexpectedly "pass" in comparison mode mode when
+ // comparing the two failures, which is not a useful result.
+ if (
+ expected_status === EXPECTED_FAIL ||
+ expected_status === EXPECTED_RANDOM
+ ) {
+ skip = true;
+ }
+ }
+
+ AddTestItem(
+ {
+ type,
+ expected: expected_status,
+ manifest: aURL.spec,
+ manifestID: TestIdentifier(aURL.spec, aManifestID),
+ allowSilentFail: allow_silent_fail,
+ minAsserts,
+ maxAsserts,
+ needsFocus: needs_focus,
+ slow,
+ skip,
+ prefSettings1: testPrefSettings,
+ prefSettings2: refPrefSettings,
+ fuzzyMinDelta: fuzzy_delta.min,
+ fuzzyMaxDelta: fuzzy_delta.max,
+ fuzzyMinPixels: fuzzy_pixels.min,
+ fuzzyMaxPixels: fuzzy_pixels.max,
+ runHttp,
+ httpDepth,
+ url1: items[1],
+ url2: items[2],
+ chaosMode,
+ wrCapture,
+ noAutoFuzz,
+ },
+ aFilter,
+ aManifestID
+ );
+ } else {
+ throw new Error(
+ "Error in manifest file " +
+ aURL.spec +
+ " line " +
+ lineNo +
+ ": unknown test type " +
+ items[0]
+ );
+ }
+ }
+}
+
+// Read all available data from an input stream and return it
+// as a string.
+function getStreamContent(inputStream) {
+ var streamBuf = "";
+ var sis = Cc["@mozilla.org/scriptableinputstream;1"].createInstance(
+ Ci.nsIScriptableInputStream
+ );
+ sis.init(inputStream);
+
+ var available;
+ while ((available = sis.available()) != 0) {
+ streamBuf += sis.read(available);
+ }
+
+ return streamBuf;
+}
+
+// Build the sandbox for fails-if(), etc., condition evaluation.
+function BuildConditionSandbox(aURL) {
+ var sandbox = new Cu.Sandbox(aURL.spec);
+ sandbox.isDebugBuild = g.debug.isDebugBuild;
+ sandbox.isCoverageBuild = g.isCoverageBuild;
+
+ sandbox.xulRuntime = Cu.cloneInto(
+ {
+ widgetToolkit: Services.appinfo.widgetToolkit,
+ OS: Services.appinfo.OS,
+ XPCOMABI: Services.appinfo.XPCOMABI,
+ },
+ sandbox
+ );
+
+ sandbox.smallScreen = false;
+ if (
+ g.containingWindow.innerWidth < 800 ||
+ g.containingWindow.innerHeight < 1000
+ ) {
+ sandbox.smallScreen = true;
+ }
+
+ var gfxInfo =
+ NS_GFXINFO_CONTRACTID in Cc &&
+ Cc[NS_GFXINFO_CONTRACTID].getService(Ci.nsIGfxInfo);
+ let readGfxInfo = function (obj, key) {
+ if (g.contentGfxInfo && key in g.contentGfxInfo) {
+ return g.contentGfxInfo[key];
+ }
+ return obj[key];
+ };
+
+ try {
+ sandbox.d2d = readGfxInfo(gfxInfo, "D2DEnabled");
+ sandbox.dwrite = readGfxInfo(gfxInfo, "DWriteEnabled");
+ sandbox.embeddedInFirefoxReality = readGfxInfo(
+ gfxInfo,
+ "EmbeddedInFirefoxReality"
+ );
+ } catch (e) {
+ sandbox.d2d = false;
+ sandbox.dwrite = false;
+ sandbox.embeddedInFirefoxReality = false;
+ }
+
+ var canvasBackend = readGfxInfo(gfxInfo, "AzureCanvasBackend");
+ var contentBackend = readGfxInfo(gfxInfo, "AzureContentBackend");
+
+ sandbox.gpuProcess = gfxInfo.usingGPUProcess;
+ sandbox.azureCairo = canvasBackend == "cairo";
+ sandbox.azureSkia = canvasBackend == "skia";
+ sandbox.skiaContent = contentBackend == "skia";
+ sandbox.azureSkiaGL = false;
+ // true if we are using the same Azure backend for rendering canvas and content
+ sandbox.contentSameGfxBackendAsCanvas =
+ contentBackend == canvasBackend ||
+ (contentBackend == "none" && canvasBackend == "cairo");
+
+ try {
+ var windowProtocol = readGfxInfo(gfxInfo, "windowProtocol");
+ sandbox.wayland = windowProtocol == "wayland";
+ } catch (e) {
+ sandbox.wayland = false;
+ }
+
+ sandbox.remoteCanvas =
+ Services.prefs.getBoolPref("gfx.canvas.remote") &&
+ sandbox.d2d &&
+ sandbox.gpuProcess;
+
+ sandbox.layersGPUAccelerated = g.windowUtils.layerManagerType != "Basic";
+ sandbox.d3d11 = g.windowUtils.layerManagerType == "Direct3D 11";
+ sandbox.d3d9 = g.windowUtils.layerManagerType == "Direct3D 9";
+ sandbox.layersOpenGL = g.windowUtils.layerManagerType == "OpenGL";
+ sandbox.swgl = g.windowUtils.layerManagerType.startsWith(
+ "WebRender (Software"
+ );
+ sandbox.layersOMTC = !!g.windowUtils.layerManagerRemote;
+
+ // Shortcuts for widget toolkits.
+ sandbox.Android = Services.appinfo.OS == "Android";
+ sandbox.cocoaWidget = Services.appinfo.widgetToolkit == "cocoa";
+ sandbox.gtkWidget = Services.appinfo.widgetToolkit == "gtk";
+ sandbox.qtWidget = Services.appinfo.widgetToolkit == "qt";
+ sandbox.winWidget = Services.appinfo.widgetToolkit == "windows";
+
+ sandbox.is64Bit = Services.appinfo.is64Bit;
+
+ // Use this to annotate reftests that fail in drawSnapshot, but
+ // the reason hasn't been investigated (or fixed) yet.
+ sandbox.useDrawSnapshot = g.useDrawSnapshot;
+ // Use this to annotate reftests that use functionality
+ // that isn't available to drawSnapshot (like any sort of
+ // compositor feature such as async scrolling).
+ sandbox.unsupportedWithDrawSnapshot = g.useDrawSnapshot;
+
+ sandbox.retainedDisplayList =
+ Services.prefs.getBoolPref("layout.display-list.retain") &&
+ !sandbox.useDrawSnapshot;
+
+ // Needed to specifically test the new and old behavior. This will eventually be removed.
+ sandbox.retainedDisplayListNew =
+ sandbox.retainedDisplayList &&
+ Services.prefs.getBoolPref("layout.display-list.retain.sc");
+
+ // GeckoView is currently uniquely identified by "android + e10s" but
+ // we might want to make this condition more precise in the future.
+ sandbox.geckoview = sandbox.Android && g.browserIsRemote;
+
+ // Scrollbars that are semi-transparent. See bug 1169666.
+ sandbox.transparentScrollbars = Services.appinfo.widgetToolkit == "gtk";
+
+ if (sandbox.Android) {
+ sandbox.AndroidVersion = Services.sysinfo.getPropertyAsInt32("version");
+
+ sandbox.emulator = readGfxInfo(gfxInfo, "adapterDeviceID").includes(
+ "Android Emulator"
+ );
+ sandbox.device = !sandbox.emulator;
+ }
+
+ // Some reftests need extra fuzz on the Android 13 Pixel 5 devices.
+ sandbox.Android13 = sandbox.AndroidVersion == "33";
+
+ sandbox.MinGW =
+ sandbox.winWidget && Services.sysinfo.getPropertyAsBool("isMinGW");
+
+ sandbox.AddressSanitizer = AppConstants.ASAN;
+ sandbox.ThreadSanitizer = AppConstants.TSAN;
+ sandbox.webrtc = AppConstants.MOZ_WEBRTC;
+ sandbox.jxl = AppConstants.MOZ_JXL;
+
+ sandbox.compareRetainedDisplayLists = g.compareRetainedDisplayLists;
+
+ sandbox.release_or_beta = AppConstants.RELEASE_OR_BETA;
+
+ var hh = Cc[NS_NETWORK_PROTOCOL_CONTRACTID_PREFIX + "http"].getService(
+ Ci.nsIHttpProtocolHandler
+ );
+ var httpProps = [
+ "userAgent",
+ "appName",
+ "appVersion",
+ "vendor",
+ "vendorSub",
+ "product",
+ "productSub",
+ "oscpu",
+ "language",
+ "misc",
+ ];
+ sandbox.http = new sandbox.Object();
+ httpProps.forEach(x => (sandbox.http[x] = hh[x]));
+
+ // Set OSX to be the Mac OS X version, as an integer, or undefined
+ // for other platforms. The integer is formed by 100 times the
+ // major version plus the minor version, so 1006 for 10.6, 1010 for
+ // 10.10, etc.
+ var osxmatch = /Mac OS X (\d+).(\d+)$/.exec(hh.oscpu);
+ sandbox.OSX = osxmatch
+ ? parseInt(osxmatch[1]) * 100 + parseInt(osxmatch[2])
+ : undefined;
+
+ // config specific prefs
+ sandbox.appleSilicon = Services.prefs.getBoolPref(
+ "sandbox.apple_silicon",
+ false
+ );
+
+ sandbox.gpuProcessForceEnabled = Services.prefs.getBoolPref(
+ "layers.gpu-process.force-enabled",
+ false
+ );
+
+ sandbox.prefs = Cu.cloneInto(
+ {
+ getBoolPref(p) {
+ return Services.prefs.getBoolPref(p);
+ },
+ getIntPref(p) {
+ return Services.prefs.getIntPref(p);
+ },
+ },
+ sandbox,
+ { cloneFunctions: true }
+ );
+
+ // Tests shouldn't care about this except for when they need to
+ // crash the content process
+ sandbox.browserIsRemote = g.browserIsRemote;
+ sandbox.browserIsFission = g.browserIsFission;
+
+ try {
+ sandbox.asyncPan =
+ g.containingWindow.docShell.asyncPanZoomEnabled &&
+ !sandbox.useDrawSnapshot;
+ } catch (e) {
+ sandbox.asyncPan = false;
+ }
+
+ // Graphics features
+ sandbox.usesRepeatResampling = sandbox.d2d;
+
+ // Running in a test-verify session?
+ sandbox.verify = Services.prefs.getBoolPref("reftest.verify", false);
+
+ // Running with a variant enabled?
+ sandbox.fission = Services.appinfo.fissionAutostart;
+ sandbox.serviceWorkerE10s = true;
+
+ if (!g.dumpedConditionSandbox) {
+ g.logger.info(
+ "Dumping representation of sandbox which can be used for expectation annotations"
+ );
+ for (let entry of Object.entries(Cu.waiveXrays(sandbox)).sort((a, b) =>
+ a[0].localeCompare(b[0])
+ )) {
+ let value =
+ typeof entry[1] === "object" ? JSON.stringify(entry[1]) : entry[1];
+ g.logger.info(` ${entry[0]}: ${value}`);
+ }
+ g.dumpedConditionSandbox = true;
+ }
+
+ return sandbox;
+}
+
+function AddRetainedDisplayListTestPrefs(
+ aSandbox,
+ aTestPrefSettings,
+ aRefPrefSettings
+) {
+ AddPrefSettings(
+ "test-",
+ "layout.display-list.retain",
+ "true",
+ aSandbox,
+ aTestPrefSettings,
+ aRefPrefSettings
+ );
+ AddPrefSettings(
+ "ref-",
+ "layout.display-list.retain",
+ "false",
+ aSandbox,
+ aTestPrefSettings,
+ aRefPrefSettings
+ );
+}
+
+function AddPrefSettings(
+ aWhere,
+ aPrefName,
+ aPrefValExpression,
+ aSandbox,
+ aTestPrefSettings,
+ aRefPrefSettings
+) {
+ var prefVal = Cu.evalInSandbox("(" + aPrefValExpression + ")", aSandbox);
+ var prefType;
+ var valType = typeof prefVal;
+ if (valType == "boolean") {
+ prefType = PREF_BOOLEAN;
+ } else if (valType == "string") {
+ prefType = PREF_STRING;
+ } else if (valType == "number" && parseInt(prefVal) == prefVal) {
+ prefType = PREF_INTEGER;
+ } else {
+ return false;
+ }
+ var setting = { name: aPrefName, type: prefType, value: prefVal };
+
+ if (
+ g.compareRetainedDisplayLists &&
+ aPrefName != "layout.display-list.retain"
+ ) {
+ // ref-pref() is ignored, test-pref() and pref() are added to both
+ if (aWhere != "ref-") {
+ aTestPrefSettings.push(setting);
+ aRefPrefSettings.push(setting);
+ }
+ } else {
+ if (aWhere != "ref-") {
+ aTestPrefSettings.push(setting);
+ }
+ if (aWhere != "test-") {
+ aRefPrefSettings.push(setting);
+ }
+ }
+ return true;
+}
+
+function ExtractRange(matches, startIndex) {
+ return {
+ min: Number(matches[startIndex]),
+ max: Number(matches[startIndex + 1]),
+ };
+}
+
+function ServeTestBase(aURL, depth) {
+ var listURL = aURL.QueryInterface(Ci.nsIFileURL);
+ var directory = listURL.file.parent;
+
+ // Allow serving a tree that's an ancestor of the directory containing
+ // the files so that they can use resources in ../ (etc.).
+ var dirPath = "/";
+ while (depth > 0) {
+ dirPath = "/" + directory.leafName + dirPath;
+ directory = directory.parent;
+ --depth;
+ }
+
+ g.count++;
+ var path = "/" + Date.now() + "/" + g.count;
+ g.server.registerDirectory(path + "/", directory);
+ // this one is needed so tests can use example.org urls for cross origin testing
+ g.server.registerDirectory("/", directory);
+
+ return g.ioService.newURI(
+ "http://localhost:" + g.httpServerPort + path + dirPath
+ );
+}
+
+export function CreateUrls(test) {
+ let manifestURL = g.ioService.newURI(test.manifest);
+
+ let testbase = manifestURL;
+ if (test.runHttp) {
+ testbase = ServeTestBase(manifestURL, test.httpDepth);
+ }
+
+ let testbasePrincipal = Services.scriptSecurityManager.createContentPrincipal(
+ testbase,
+ {}
+ );
+ Services.perms.addFromPrincipal(
+ testbasePrincipal,
+ "allowXULXBL",
+ Services.perms.ALLOW_ACTION
+ );
+
+ function FileToURI(file) {
+ if (file === null) {
+ return file;
+ }
+
+ var testURI = g.ioService.newURI(file, null, testbase);
+ let isChromeOrViewSource =
+ testURI.scheme == "chrome" || testURI.scheme == "view-source";
+ let principal = isChromeOrViewSource
+ ? Services.scriptSecurityManager.getSystemPrincipal()
+ : Services.scriptSecurityManager.createContentPrincipal(manifestURL, {});
+ Services.scriptSecurityManager.checkLoadURIWithPrincipal(
+ principal,
+ testURI,
+ Ci.nsIScriptSecurityManager.DISALLOW_SCRIPT
+ );
+ return testURI;
+ }
+
+ let files = [test.url1, test.url2];
+ [test.url1, test.url2] = files.map(FileToURI);
+
+ return test;
+}
+
+function TestIdentifier(aUrl, aManifestID) {
+ // Construct a platform-independent and location-independent test identifier for
+ // a url; normally the identifier looks like a posix-compliant relative file
+ // path.
+ // Test urls may be simple file names, chrome: urls with full paths, about:blank, etc.
+ if (aUrl.startsWith("about:") || aUrl.startsWith("data:")) {
+ return aUrl;
+ }
+ var pos = aUrl.lastIndexOf("/");
+ var url = pos < 0 ? aUrl : aUrl.substring(pos + 1);
+ return aManifestID + "/" + url;
+}
+
+function AddTestItem(aTest, aFilter, aManifestID) {
+ if (!aFilter) {
+ aFilter = [null, [], false];
+ }
+
+ var identifier = TestIdentifier(aTest.url1, aManifestID);
+ if (aTest.url2 !== null) {
+ identifier = [
+ identifier,
+ aTest.type,
+ TestIdentifier(aTest.url2, aManifestID),
+ ];
+ }
+
+ var { url1, url2 } = CreateUrls(Object.assign({}, aTest));
+
+ var globalFilter = aFilter[0];
+ var manifestFilter = aFilter[1];
+ var invertManifest = aFilter[2];
+ if (globalFilter && !globalFilter.test(url1.spec)) {
+ if (url2 === null) {
+ return;
+ }
+ if (globalFilter && !globalFilter.test(url2.spec)) {
+ return;
+ }
+ }
+ if (manifestFilter && !(invertManifest ^ manifestFilter.test(url1.spec))) {
+ if (url2 === null) {
+ return;
+ }
+ if (manifestFilter && !(invertManifest ^ manifestFilter.test(url2.spec))) {
+ return;
+ }
+ }
+ if (
+ g.focusFilterMode == FOCUS_FILTER_NEEDS_FOCUS_TESTS &&
+ !aTest.needsFocus
+ ) {
+ return;
+ }
+ if (
+ g.focusFilterMode == FOCUS_FILTER_NON_NEEDS_FOCUS_TESTS &&
+ aTest.needsFocus
+ ) {
+ return;
+ }
+
+ aTest.identifier = identifier;
+ g.urls.push(aTest);
+ // Periodically log progress to avoid no-output timeout on slow platforms.
+ // No-output timeouts during manifest parsing have been a problem for
+ // jsreftests on Android/debug. Any logging resets the no-output timer,
+ // even debug logging which is normally not displayed.
+ if (g.urls.length % 5000 == 0) {
+ g.logger.debug(g.urls.length + " tests found...");
+ }
+}