232 lines
7.5 KiB
JavaScript
232 lines
7.5 KiB
JavaScript
/* -*- 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 { FileUtils } = ChromeUtils.importESModule(
|
|
"resource://gre/modules/FileUtils.sys.mjs"
|
|
);
|
|
|
|
// The xpcshell test harness sets PYTHON so we can read it here.
|
|
var gPythonName = Services.env.get("PYTHON");
|
|
|
|
const gCwd = Services.dirsvc.get("CurWorkD", Ci.nsIFile);
|
|
|
|
function getRelativeFile(...components) {
|
|
return new FileUtils.File(PathUtils.join(gCwd.path, ...components));
|
|
}
|
|
|
|
// If we're testing locally, the executable file is in "CurProcD". Otherwise,
|
|
// it is in another location that we have to find.
|
|
function getExecutable(aFilename) {
|
|
let file = new FileUtils.File(
|
|
PathUtils.join(Services.dirsvc.get("CurProcD", Ci.nsIFile).path, aFilename)
|
|
);
|
|
if (!file.exists()) {
|
|
file = gCwd.clone();
|
|
while (file.path.includes("xpcshell")) {
|
|
file = file.parent;
|
|
}
|
|
file.append("bin");
|
|
file.append(aFilename);
|
|
}
|
|
return file;
|
|
}
|
|
|
|
var gIsWindows = Services.appinfo.OS === "WINNT";
|
|
var gDmdTestFile = getExecutable("SmokeDMD" + (gIsWindows ? ".exe" : ""));
|
|
|
|
var gDmdScriptFile = getExecutable("dmd.py");
|
|
|
|
var gScanTestFile = getRelativeFile("scan-test.py");
|
|
|
|
function readFile(aFile) {
|
|
let fstream = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(
|
|
Ci.nsIFileInputStream
|
|
);
|
|
let cstream = Cc["@mozilla.org/intl/converter-input-stream;1"].createInstance(
|
|
Ci.nsIConverterInputStream
|
|
);
|
|
fstream.init(aFile, -1, 0, 0);
|
|
cstream.init(fstream, "UTF-8", 0, 0);
|
|
|
|
let data = "";
|
|
let str = {};
|
|
let read = 0;
|
|
do {
|
|
// Read as much as we can and put it in str.value.
|
|
read = cstream.readString(0xffffffff, str);
|
|
data += str.value;
|
|
} while (read != 0);
|
|
|
|
cstream.close(); // this closes fstream
|
|
return data.replace(/\r/g, ""); // normalize line endings
|
|
}
|
|
|
|
function runProcess(aExeFile, aArgs) {
|
|
let process = Cc["@mozilla.org/process/util;1"].createInstance(Ci.nsIProcess);
|
|
process.init(aExeFile);
|
|
process.run(/* blocking = */ true, aArgs, aArgs.length);
|
|
return process.exitValue;
|
|
}
|
|
|
|
function test(aPrefix, aArgs) {
|
|
// DMD writes the JSON files to CurWorkD, so we do likewise here with
|
|
// |actualFile| for consistency. It is removed once we've finished.
|
|
let expectedFile = getRelativeFile(`${aPrefix}-expected.txt`);
|
|
let actualFile = getRelativeFile(`${aPrefix}-actual.txt`);
|
|
|
|
// Run dmd.py on the JSON file, producing |actualFile|.
|
|
|
|
let args = [
|
|
gDmdScriptFile.path,
|
|
"--filter-stacks-for-testing",
|
|
"-o",
|
|
actualFile.path,
|
|
].concat(aArgs);
|
|
|
|
runProcess(new FileUtils.File(gPythonName), args);
|
|
|
|
// Compare |expectedFile| with |actualFile|. We produce nice diffs with
|
|
// /usr/bin/diff on systems that have it (Mac and Linux). Otherwise (Windows)
|
|
// we do a string compare of the file contents and then print them both if
|
|
// they don't match.
|
|
|
|
let success;
|
|
try {
|
|
let rv = runProcess(new FileUtils.File("/usr/bin/diff"), [
|
|
"-u",
|
|
expectedFile.path,
|
|
actualFile.path,
|
|
]);
|
|
success = rv == 0;
|
|
} catch (e) {
|
|
let expectedData = readFile(expectedFile);
|
|
let actualData = readFile(actualFile);
|
|
success = expectedData === actualData;
|
|
if (!success) {
|
|
expectedData = expectedData.split("\n");
|
|
actualData = actualData.split("\n");
|
|
for (let i = 0; i < expectedData.length; i++) {
|
|
print("EXPECTED:" + expectedData[i]);
|
|
}
|
|
for (let i = 0; i < actualData.length; i++) {
|
|
print(" ACTUAL:" + actualData[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
ok(success, aPrefix);
|
|
|
|
actualFile.remove(true);
|
|
}
|
|
|
|
// Run scan-test.py on the JSON file and see if it succeeds.
|
|
function scanTest(aJsonFilePath, aExtraArgs) {
|
|
let args = [gScanTestFile.path, aJsonFilePath].concat(aExtraArgs);
|
|
|
|
return runProcess(new FileUtils.File(gPythonName), args) == 0;
|
|
}
|
|
|
|
function run_test() {
|
|
let jsonFile, jsonFile2;
|
|
|
|
// These tests do complete end-to-end testing of DMD, i.e. both the C++ code
|
|
// that generates the JSON output, and the script that post-processes that
|
|
// output.
|
|
//
|
|
// Run these synchronously, because test() updates the complete*.json files
|
|
// in-place (to fix stacks) when it runs dmd.py, and that's not safe to do
|
|
// asynchronously.
|
|
|
|
Services.env.set("DMD", "1");
|
|
|
|
runProcess(gDmdTestFile, []);
|
|
|
|
function test2(aTestName, aMode) {
|
|
let name = "complete-" + aTestName + "-" + aMode;
|
|
jsonFile = getRelativeFile(`${name}.json`);
|
|
test(name, [jsonFile.path]);
|
|
jsonFile.remove(true);
|
|
}
|
|
|
|
// Please keep this in sync with RunTests() in SmokeDMD.cpp.
|
|
|
|
test2("empty", "live");
|
|
test2("empty", "dark-matter");
|
|
test2("empty", "cumulative");
|
|
|
|
test2("full1", "live");
|
|
test2("full1", "dark-matter");
|
|
|
|
test2("full2", "dark-matter");
|
|
test2("full2", "cumulative");
|
|
|
|
test2("partial", "live");
|
|
|
|
// Heap scan testing.
|
|
jsonFile = getRelativeFile("basic-scan.json");
|
|
ok(scanTest(jsonFile.path), "Basic scan test");
|
|
|
|
let is64Bit = Services.appinfo.is64Bit;
|
|
let basicScanFileName = "basic-scan-" + (is64Bit ? "64" : "32");
|
|
test(basicScanFileName, ["--clamp-contents", jsonFile.path]);
|
|
ok(
|
|
scanTest(jsonFile.path, ["--clamp-contents"]),
|
|
"Scan with address clamping"
|
|
);
|
|
|
|
// Run the generic test a second time to ensure that the first time produced
|
|
// valid JSON output. "--clamp-contents" is passed in so we don't have to have
|
|
// more variants of the files.
|
|
test(basicScanFileName, ["--clamp-contents", jsonFile.path]);
|
|
jsonFile.remove(true);
|
|
|
|
// These tests only test the post-processing script. They use hand-written
|
|
// JSON files as input. Ideally the JSON files would contain comments
|
|
// explaining how they work, but JSON doesn't allow comments, so I've put
|
|
// explanations here.
|
|
|
|
// This just tests that stack traces of various lengths are truncated
|
|
// appropriately. The number of records in the output is different for each
|
|
// of the tested values.
|
|
jsonFile = getRelativeFile("script-max-frames.json");
|
|
test("script-max-frames-8", [jsonFile.path]); // --max-frames=8 is the default
|
|
test("script-max-frames-3", [
|
|
"--max-frames=3",
|
|
"--no-fix-stacks",
|
|
jsonFile.path,
|
|
]);
|
|
test("script-max-frames-1", ["--max-frames=1", jsonFile.path]);
|
|
|
|
// This file has three records that are shown in a different order for each
|
|
// of the different sort values. It also tests the handling of gzipped JSON
|
|
// files.
|
|
jsonFile = getRelativeFile("script-sort-by.json.gz");
|
|
test("script-sort-by-usable", ["--sort-by=usable", jsonFile.path]);
|
|
test("script-sort-by-req", [
|
|
"--sort-by=req",
|
|
"--no-fix-stacks",
|
|
jsonFile.path,
|
|
]);
|
|
test("script-sort-by-slop", ["--sort-by=slop", jsonFile.path]);
|
|
test("script-sort-by-num-blocks", ["--sort-by=num-blocks", jsonFile.path]);
|
|
|
|
// This file has several real stack traces taken from Firefox execution, each
|
|
// of which tests a different allocator function (or functions).
|
|
jsonFile = getRelativeFile("script-ignore-alloc-fns.json");
|
|
test("script-ignore-alloc-fns", ["--ignore-alloc-fns", jsonFile.path]);
|
|
|
|
// This tests "live"-mode diffs.
|
|
jsonFile = getRelativeFile("script-diff-live1.json");
|
|
jsonFile2 = getRelativeFile("script-diff-live2.json");
|
|
test("script-diff-live", [jsonFile.path, jsonFile2.path]);
|
|
|
|
// This tests "dark-matter"-mode diffs.
|
|
jsonFile = getRelativeFile("script-diff-dark-matter1.json");
|
|
jsonFile2 = getRelativeFile("script-diff-dark-matter2.json");
|
|
test("script-diff-dark-matter", [jsonFile.path, jsonFile2.path]);
|
|
}
|