summaryrefslogtreecommitdiffstats
path: root/PerformanceTests/SunSpider/sunspider
diff options
context:
space:
mode:
Diffstat (limited to 'PerformanceTests/SunSpider/sunspider')
-rwxr-xr-xPerformanceTests/SunSpider/sunspider244
1 files changed, 244 insertions, 0 deletions
diff --git a/PerformanceTests/SunSpider/sunspider b/PerformanceTests/SunSpider/sunspider
new file mode 100755
index 0000000..dbe26cd
--- /dev/null
+++ b/PerformanceTests/SunSpider/sunspider
@@ -0,0 +1,244 @@
+#!/usr/bin/perl -w
+
+# Copyright (C) 2007 Apple Inc. All rights reserved.
+# Copyright (C) 2007 Eric Seidel <eric@webkit.org>
+#
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions
+# are met:
+# 1. Redistributions of source code must retain the above copyright
+# notice, this list of conditions and the following disclaimer.
+# 2. Redistributions in binary form must reproduce the above copyright
+# notice, this list of conditions and the following disclaimer in the
+# documentation and/or other materials provided with the distribution.
+#
+# THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
+# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
+# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+# PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+# OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+use strict;
+use Getopt::Long;
+use File::Basename;
+use File::Spec;
+use Cwd;
+use POSIX qw(strftime);
+use Time::HiRes qw(gettimeofday tv_interval);
+
+my $showHelp = 0;
+my $runShark = 0;
+my $runShark20 = 0;
+my $runSharkCache = 0;
+my $ubench = 0;
+my $v8suite = 0;
+my $suite = "";
+my $parseOnly = 0;
+my $jsShellPath;
+my $jsShellArgs = "";
+my $setBaseline = 0;
+my $testsPattern;
+my $testRuns = 10;
+
+my $programName = basename($0);
+my $usage = <<EOF;
+Usage: $programName --shell=[path] [options]
+ --help Show this help message
+ --set-baseline Set baseline for future comparisons
+ --shell Path to JavaScript shell
+ --args Arguments to pass to JavaScript shell
+ --runs Number of times to run tests (default: $testRuns)
+ --tests Only run tests matching provided pattern
+ --shark Sample execution time with the Mac OS X "Shark" performance testing tool (implies --runs=1)
+ --shark20 Like --shark, but with a 20 microsecond sampling interval
+ --shark-cache Like --shark, but performs a L2 cache-miss sample instead of time sample
+ --suite Select a specific benchmark suite. The default is sunspider-0.9.1
+ --ubench Use microbenchmark suite instead of regular tests. Same as --suite=ubench
+ --v8-suite Use the V8 benchmark suite. Same as --suite=v8-v4
+ --parse-only Use the parse-only benchmark suite. Same as --suite=parse-only
+EOF
+
+GetOptions('runs=i' => \$testRuns,
+ 'set-baseline' => \$setBaseline,
+ 'shell=s' => \$jsShellPath,
+ 'args=s' => \$jsShellArgs,
+ 'shark' => \$runShark,
+ 'shark20' => \$runShark20,
+ 'shark-cache' => \$runSharkCache,
+ 'suite=s' => \$suite,
+ 'ubench' => \$ubench,
+ 'v8-suite' => \$v8suite,
+ 'parse-only' => \$parseOnly,
+ 'tests=s' => \$testsPattern,
+ 'help' => \$showHelp);
+
+
+$suite = "ubench" if ($ubench);
+$suite = "v8-v4" if ($v8suite);
+$suite = "parse-only" if ($parseOnly);
+$suite = "sunspider-0.9.1" if (!$suite);
+
+my $resultDirectory = "${suite}-results";
+
+my $suitePath = $suite;
+$suitePath = "tests/" . $suitePath unless ($suite =~ /\//);
+
+$runShark = 1 if $runSharkCache;
+$runShark = 20 if $runShark20;
+$testRuns = 1 if $runShark;
+if ($runShark && ! -x "/usr/bin/shark") {
+ die "Please install CHUD tools from http://developer.apple.com/tools/download/\n";
+}
+
+my $sharkCacheProfileIndex = 0;
+if ($runSharkCache) {
+ my $sharkProfileList = `shark -l 2>&1`;
+ for my $profile (split(/\n/, $sharkProfileList)) {
+ $profile =~ /(\d+) - (.+)/;
+ next unless (defined $1);
+ my $profileIndex = $1;
+ my $profileName = $2;
+ if ($profileName =~ /L2 Cache/) {
+ $sharkCacheProfileIndex = $profileIndex;
+ print "Using Shark L2 Cache Miss Profile: " . $profile . "\n";
+ last;
+ }
+ }
+ die "Failed to find L2 Cache Miss Profile for --shark-cache\n" unless ($sharkCacheProfileIndex);
+}
+
+if (!$jsShellPath || $showHelp) {
+ print STDERR $usage;
+ exit 1;
+}
+
+sub dumpToFile($$)
+{
+ my ($contents, $path) = @_;
+ open FILE, ">", $path or die "Failed to open $path";
+ print FILE $contents;
+ close FILE;
+}
+
+my @tests = ();
+my @categories = ();
+my %uniqueCategories = ();
+
+sub loadTestsList()
+{
+ open TESTLIST, "<", "${suitePath}/LIST" or die "Can't find ${suitePath}/LIST";
+ while (<TESTLIST>) {
+ chomp;
+ next unless !$testsPattern || /$testsPattern/;
+
+ push @tests, $_;
+ my $category = $_;
+ $category =~ s/-.*//;
+ if (!$uniqueCategories{$category}) {
+ push @categories, $category;
+ $uniqueCategories{$category} = $category;
+ }
+ }
+ close TESTLIST;
+}
+
+my $timeString = strftime "%Y-%m-%d-%H.%M.%S", localtime $^T;
+my $prefixFile = "$resultDirectory/sunspider-test-prefix.js";
+my $resultsFile = "$resultDirectory/sunspider-results-$timeString.js";
+
+sub writePrefixFile()
+{
+ my $prefix = "var suitePath = " . '"' . $suitePath . '"' . ";\n";
+ $prefix .= "var tests = [ " . join(", ", map { '"' . $_ . '"' } @tests) . " ];\n";
+ $prefix .= "var categories = [ " . join(", ", map { '"' . $_ . '"' } @categories) . " ];\n";
+
+ mkdir "$resultDirectory";
+ dumpToFile($prefix, $prefixFile);
+}
+
+sub runTestsOnce($)
+{
+ my ($useShark) = @_;
+ my $shellArgs = $jsShellArgs . " -f $prefixFile -f resources/sunspider-standalone-driver.js 2> " . File::Spec->devnull();
+ my $output;
+ if ($useShark) {
+ my $intervalArg = $useShark == 20 ? "-I 20u" : "";
+ my $cacheArg = $runSharkCache ? "-c $sharkCacheProfileIndex" : "";
+ $output = `shark $intervalArg $cacheArg -i -1-q "$jsShellPath" $shellArgs`;
+ } else {
+ $output = `"$jsShellPath" $shellArgs | grep -v break`;
+ }
+ return $output;
+}
+
+sub newestFile($$)
+{
+ my ($dir, $pattern) = @_;
+
+ my $newestAge;
+ my $newestFile = "";
+ opendir DIR, $dir or die;
+ for my $file (readdir DIR) {
+ if ($file =~ $pattern) {
+ my $age = -M "$dir/$file";
+ if (!defined $newestAge || $age < $newestAge) {
+ $newestFile = $file;
+ $newestAge = $age;
+ }
+ }
+ }
+ closedir DIR;
+
+ return "$dir/$newestFile";
+}
+
+loadTestsList();
+if ($testsPattern) {
+ print STDERR "Found " . scalar(@tests) . " tests matching '" . $testsPattern . "'\n";
+} else {
+ print STDERR "Found " . scalar(@tests) . " tests\n";
+}
+die "No tests to run" unless scalar(@tests);
+print STDERR "Running SunSpider once for warmup, then " . ($runShark ? "under Shark" : "$testRuns time" . ($testRuns == 1 ? "" : "s")) . "\n";
+writePrefixFile();
+
+runTestsOnce(0);
+print "Discarded first run.\n";
+
+my $result;
+my $count = 0;
+my @results = ();
+my $total = 0;
+print "[";
+while ($count++ < $testRuns) {
+ $result = runTestsOnce($runShark);
+ $result =~ s/\r\n/\n/g;
+ chomp $result;
+ push @results, $result;
+ print $result;
+ print ",\n" unless ($count == $testRuns);
+}
+print "]\n";
+
+my $output = "var output = [\n" . join(",\n", @results) . "\n];\n";
+dumpToFile($output, $resultsFile);
+dumpToFile(File::Spec->rel2abs($resultsFile), "$resultDirectory/baseline-filename.txt") if $setBaseline;
+
+system("$jsShellPath", "-f", $prefixFile, "-f", $resultsFile, "-f", "resources/sunspider-analyze-results.js");
+
+print("\nResults are located at $resultsFile\n");
+
+if ($runShark) {
+ my $newestMShark = newestFile(".", qr/\.mshark$/);
+ if ($newestMShark) {
+ my $profileFile = "$resultDirectory/sunspider-profile-$timeString.mshark";
+ rename $newestMShark, $profileFile or die;
+ exec "/usr/bin/open", $profileFile;
+ }
+}