jdk-24/nashorn/test/script/basic/run-octane.js

306 lines
8.8 KiB
JavaScript
Raw Normal View History

/*
* Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/**
* @subtest
*/
function initZlib() {
zlib = new BenchmarkSuite('zlib', [152815148], [
new Benchmark('zlib', false, true, 10,
runZlib, undefined, tearDownZlib, null, 3)]);
}
var tests = [
{name:"box2d", files:["box2d.js"], suite:"Box2DBenchmark"},
{name:"code-load", files:["code-load.js"], suite:"CodeLoad"},
{name:"crypto", files:["crypto.js"], suite:"Crypto"},
{name:"deltablue", files:["deltablue.js"], suite:"DeltaBlue"},
{name:"earley-boyer", files:["earley-boyer.js"], suite:"EarleyBoyer"},
{name:"gbemu", files:["gbemu-part1.js", "gbemu-part2.js"], suite:"GameboyBenchmark"},
{name:"mandreel", files:["mandreel.js"], suite:"MandreelBenchmark"},
{name:"navier-stokes", files:["navier-stokes.js"], suite:"NavierStokes"},
{name:"pdfjs", files:["pdfjs.js"], suite:"PdfJS"},
{name:"raytrace", files:["raytrace.js"], suite:"RayTrace"},
{name:"regexp", files:["regexp.js"], suite:"RegExpSuite"},
{name:"richards", files:["richards.js"], suite:"Richards"},
{name:"splay", files:["splay.js"], suite:"Splay"},
{name:"typescript", files:["typescript.js", "typescript-input.js", "typescript-compiler.js"], suite:"typescript"},
//zlib currently disabled - requires read
{name:"zlib", files:["zlib.js", "zlib-data.js"], suite:"zlib", before:initZlib}
];
var dir = (typeof(__DIR__) == 'undefined') ? "test/script/basic/" : __DIR__;
// TODO: why is this path hard coded when it's defined in project properties?
var path = dir + "../external/octane/";
var runtime = undefined;
var verbose = false;
var numberOfIterations = 5;
function endsWith(str, suffix) {
return str.indexOf(suffix, str.length - suffix.length) !== -1;
}
function should_compile_only(name) {
return (typeof compile_only !== 'undefined')
}
function load_bench(arg) {
for (var idx = 0; idx < arg.files.length; idx++) {
var f = arg.files[idx];
var file = f.split('/');
var file_name = path + file[file.length - 1];
var compile_and_return = should_compile_only(file_name);
if (compile_and_return) {
if (typeof compile_only === 'undefined') { //for a run, skip compile onlies, don't even compile them
return true;
}
}
print_verbose(arg, "loading '" + arg.name + "' [" + f + "]...");
load(file_name);
8007062: Split Lower up into Lower/Attr/FinalizeTypes. Integrate AccessSpecalizer into FinalizeTypes Lower suffered from being a "God class" trying to do everything at once. As Nashorn code generation has grown, so has Lower. It does several post processing passes, tries to do several things at once even though all type information isn't in place, adjusting state afterwards and so on. It also performs control flow analysis, type attribution and constant folding, and everything else code generation related before byte code emission. I have now separated the compilation process into Lower (create low level nodes from high level ones, copy code such as finally block inlining etc), Attr (assign types and symbols to all nodes - freeze slot and scope information) and FinalizeTypes (insert explicit casts, specialize invoke dynamic types for scope accesses). I've removed the kludgy AccessSpecializer, as this now integrates naturally with typing. Everything is now much easier to read and each module performs only one thing. I have added separate loggers for the separate tiers. In the process I have also fixed: (1) problems with type coercion (see test/script/basic/typecoercion.js, basically our coercion was too late and our symbol inference was erroneous. This only manifested itself in very rare occasions where toNumber coercion has side effects, such as for example when valueOf is overridden) (2) copying literal nodes (literal copy did not use the superclass copy, which made all the Node specific fields not to be copied (3) erroneous literal tokenization (literals shouldn't always just inherit token information from whatever node that creates them) (4) splitter weighnodes - unary nodes were considered weightless (4) removed the hateful and kludgy "VarNode.shouldAppend", which really isn't needed when we have an attribution phase that determines self reference symbols (the only thing it was used for) (5) duplicate line number issues in the parser (6) convert bug in CodeGenerator for intermediate results of scope accesses (see test/script/basic/access-specializer.js) ... Several of these things just stopped being problems with the new architecture "can't happen anymore" and are not bug fixes per se. All tests run. No performance regressions exist that I've been able to measure. Some increases in performance were measured, but in the statistical margin of error (which is very wide as HotSpot currently has warmup issues with LambdaForms/invoke dynamic). Compile speed has not measurably increased. Reviewed-by: jlaskey, attila
2013-01-30 12:26:45 +01:00
}
if (typeof arg.before !== 'undefined') {
arg.before();
}
if (compile_and_return) {
print_always(arg, "Compiled OK");
}
return !compile_and_return;
}
function run_one_benchmark(arg, iters) {
if (!load_bench(arg)) {
return;
}
var success = true;
var current_name;
if (iters == undefined) {
iters = numberOfIterations;
} else {
numberOfIterations = iters;
}
var benchmarks = eval(arg.suite + ".benchmarks");
var min_score = 1e9;
var max_score = 0;
var mean_score = 0;
try {
for (var x = 0; x < benchmarks.length ; x++) {
//do warmup run
//reset random number generator needed as of octane 9 before each run
BenchmarkSuite.ResetRNG();
benchmarks[x].Setup();
}
BenchmarkSuite.ResetRNG();
print_verbose(arg, "running '" + arg.name + "' for " + iters + " iterations of no less than " + min_time + " seconds");
var scores = [];
var min_time_ms = min_time * 1000;
var len = benchmarks.length;
for (var it = 0; it < iters + 1; it++) {
//every iteration must take a minimum of 10 secs
var ops = 0;
var elapsed = 0;
var start = new Date;
do {
for (var i = 0; i < len; i++) {
benchmarks[i].run();
//important - no timing here like elapsed = new Date() - start, as in the
//original harness. This will make timing very non-deterministic.
//NOTHING else must live in this loop
}
ops += len;
elapsed = new Date - start;
} while (elapsed < min_time * 1000);
var score = ops / elapsed * 1000 * 60;
scores.push(score);
var name = it == 0 ? "warmup" : "iteration " + it;
print_verbose(arg, name + " finished " + score.toFixed(0) + " ops/minute");
}
for (var x = 0; x < benchmarks.length ; x++) {
benchmarks[x].TearDown();
}
for (var x = 1; x < iters + 1 ; x++) {
mean_score += scores[x];
min_score = Math.min(min_score, scores[x]);
max_score = Math.max(max_score, scores[x]);
}
mean_score /= iters;
} catch (e) {
print_always(arg, "*** Aborted and setting score to zero. Reason: " + e);
if (e instanceof java.lang.Throwable) {
e.printStackTrace();
}
mean_score = min_score = max_score = 0;
scores = [0];
}
var res = mean_score.toFixed(0);
if (verbose) {
res += " ops/minute (" + min_score.toFixed(0) + "-" + max_score.toFixed(0) + "), warmup=" + scores[0].toFixed(0);
}
print_always(arg, res);
}
function runtime_string() {
return runtime == undefined ? "" : ("[" + runtime + "] ");
}
function print_always(arg, x) {
print(runtime_string() + "[" + arg.name + "] " + x);
}
function print_verbose(arg, x) {
if (verbose) {
print_always(arg, x)
}
}
function run_suite(tests, iters) {
for (var idx = 0; idx < tests.length; idx++) {
run_one_benchmark(tests[idx], iters);
}
}
var args = [];
if (typeof $ARGS !== 'undefined') {
args = $ARGS;
} else if (typeof arguments !== 'undefined' && arguments.length != 0) {
args = arguments;
}
var new_args = [];
for (i in args) {
if (args[i].toString().indexOf(' ') != -1) {
args[i] = args[i].replace(/\/$/, '');
var s = args[i].split(' ');
for (j in s) {
new_args.push(s[j]);
}
} else {
new_args.push(args[i]);
}
}
if (new_args.length != 0) {
args = new_args;
}
var tests_found = [];
var iters = undefined;
var min_time = 5;
for (var i = 0; i < args.length; i++) {
arg = args[i];
if (arg == "--iterations") {
iters = +args[++i];
if (isNaN(iters)) {
throw "'--iterations' must be followed by integer";
}
} else if (arg == "--runtime") {
runtime = args[++i];
} else if (arg == "--verbose") {
verbose = true;
} else if (arg == "--min-time") {
min_time = +args[++i];
if (isNaN(iters)) {
throw "'--min-time' must be followed by integer";
}
} else if (arg == "") {
continue; //skip
} else {
var found = false;
for (j in tests) {
if (tests[j].name === arg) {
tests_found.push(tests[j]);
found = true;
break;
}
}
if (!found) {
var str = "unknown test name: '" + arg + "' -- valid names are: ";
for (j in tests) {
if (j != 0) {
str += ", ";
}
str += "'" + tests[j].name + "'";
}
throw str;
}
}
}
if (tests_found.length == 0) {
for (i in tests) {
tests_found.push(tests[i]);
}
}
// returns false for rhino, v8 and all other javascript runtimes, true for Nashorn
function is_this_nashorn() {
return typeof Error.dumpStack == 'function'
}
if (is_this_nashorn()) {
try {
read = readFully;
} catch (e) {
print("ABORTING: Cannot find 'readFully'. You must have scripting enabled to use this test harness. (-scripting)");
throw e;
}
}
// run tests in alphabetical order by name
tests_found.sort(function(a, b) {
if (a.name < b.name) {
return -1;
} else if (a.name > b.name) {
return 1;
} else {
return 0;
}
});
load(path + 'base.js');
run_suite(tests_found, iters);