pegjs/benchmark/runner.js

132 lines
3.5 KiB
JavaScript
Raw Normal View History

"use strict";
2015-06-08 19:46:28 +02:00
(function(root, factory) {
if (typeof module !== 'undefined' && module.exports) {
module.exports = factory;
} else {
root.Runner = factory(root.PEG);
}
}(this, function(PEG) {
return {
run: function(benchmarks, runCount, options, callbacks) {
2015-06-08 19:46:28 +02:00
/* Queue */
2015-06-08 19:46:28 +02:00
var Q = {
functions: [],
2015-06-08 19:46:28 +02:00
add: function(f) {
this.functions.push(f);
},
2015-06-08 19:46:28 +02:00
run: function() {
if (this.functions.length > 0) {
this.functions.shift()();
2015-06-08 19:46:28 +02:00
/*
* We can't use |arguments.callee| here because |this| would get
* messed-up in that case.
*/
setTimeout(function() { Q.run(); }, 0);
}
}
2015-06-08 19:46:28 +02:00
};
/*
* The benchmark itself is factored out into several functions (some of them
* generated), which are enqueued and run one by one using |setTimeout|. We
* do this for two reasons:
*
* 1. To avoid bowser mechanism for interrupting long-running scripts to
* kick-in (or at least to not kick-in that often).
*
* 2. To ensure progressive rendering of results in the browser (some
* browsers do not render at all when running JavaScript code).
*
* The enqueued functions share state, which is all stored in the properties
* of the |state| object.
*/
var state = {}, i, j;
function initialize() {
callbacks.start();
state.totalInputSize = 0;
state.totalParseTime = 0;
}
2015-06-08 19:46:28 +02:00
function benchmarkInitializer(i) {
return function() {
callbacks.benchmarkStart(benchmarks[i]);
2015-06-08 19:46:28 +02:00
state.parser = PEG.buildParser(
callbacks.readFile("../examples/" + benchmarks[i].id + ".pegjs"),
options
);
state.benchmarkInputSize = 0;
state.benchmarkParseTime = 0;
};
}
2015-06-08 19:46:28 +02:00
function testRunner(i, j) {
return function() {
var benchmark = benchmarks[i],
test = benchmark.tests[j],
input, parseTime, averageParseTime, k, t;
2015-06-08 19:46:28 +02:00
callbacks.testStart(benchmark, test);
2015-06-08 19:46:28 +02:00
input = callbacks.readFile(benchmark.id + "/" + test.file);
2015-06-08 19:46:28 +02:00
parseTime = 0;
for (k = 0; k < runCount; k++) {
t = (new Date()).getTime();
state.parser.parse(input);
parseTime += (new Date()).getTime() - t;
}
averageParseTime = parseTime / runCount;
2015-06-08 19:46:28 +02:00
callbacks.testFinish(benchmark, test, input.length, averageParseTime);
2015-06-08 19:46:28 +02:00
state.benchmarkInputSize += input.length;
state.benchmarkParseTime += averageParseTime;
};
}
2015-06-08 19:46:28 +02:00
function benchmarkFinalizer(i) {
return function() {
callbacks.benchmarkFinish(
benchmarks[i],
state.benchmarkInputSize,
state.benchmarkParseTime
);
2015-06-08 19:46:28 +02:00
state.totalInputSize += state.benchmarkInputSize;
state.totalParseTime += state.benchmarkParseTime;
};
}
2015-06-08 19:46:28 +02:00
function finalize() {
callbacks.finish(state.totalInputSize, state.totalParseTime);
}
/* Main */
2015-06-08 19:46:28 +02:00
Q.add(initialize);
for (i = 0; i < benchmarks.length; i++) {
Q.add(benchmarkInitializer(i));
for (j = 0; j < benchmarks[i].tests.length; j++) {
Q.add(testRunner(i, j));
}
Q.add(benchmarkFinalizer(i));
}
2015-06-08 19:46:28 +02:00
Q.add(finalize);
Q.run();
}
2015-06-08 19:46:28 +02:00
};
2015-06-08 19:46:28 +02:00
}));