636ed6c3fe00691201fac5b5f2d6114cab052a83
[WebKit-https.git] / SunSpider / sunspider
1 #!/usr/bin/perl -w
2
3 # Copyright (C) 2007 Apple Inc. All rights reserved.
4 # Copyright (C) 2007 Eric Seidel <eric@webkit.org>
5 #
6 # Redistribution and use in source and binary forms, with or without
7 # modification, are permitted provided that the following conditions
8 # are met:
9 # 1. Redistributions of source code must retain the above copyright
10 #    notice, this list of conditions and the following disclaimer.
11 # 2. Redistributions in binary form must reproduce the above copyright
12 #    notice, this list of conditions and the following disclaimer in the
13 #    documentation and/or other materials provided with the distribution.
14 #
15 # THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
16 # EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17 # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
18 # PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
19 # CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
20 # EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21 # PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
22 # PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
23 # OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
25 # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
26
27 use strict;
28 use Getopt::Long;
29 use File::Basename;
30 use Cwd;
31 use POSIX qw(strftime);
32
33 my $showHelp = 0;
34 my $runShark = 0;
35 my $jsShellPath;
36 my $testsPattern;
37 my $testRuns = 5; # This number may be different from what ./sunspider defaults to (that's OK)
38
39 my $programName = basename($0);
40 my $usage = <<EOF;
41 Usage: $programName --shell=[path] [options]
42   --help        Show this help message
43   --shell       Path to javascript shell
44   --runs        Number of times to run tests (default: $testRuns)
45   --tests       Only run tests matching provided pattern
46   --shark       Sample with the Mac OS X "Shark" performance testing tool (implies --runs=1)
47 EOF
48
49 GetOptions('runs=i' => \$testRuns,
50            'shell=s' => \$jsShellPath,
51            'shark' => \$runShark,
52            'tests=s' => \$testsPattern,
53            'help' => \$showHelp);
54
55 $testRuns = 1 if $runShark;
56
57 if (!$jsShellPath || $showHelp) {
58    print STDERR $usage;
59    exit 1;
60 }
61
62 sub dumpToFile($$)
63 {
64     my ($contents, $path) = @_;
65     open FILE, ">", $path or die "Failed to open $path";
66     print FILE $contents;
67     close FILE;
68 }
69
70 my @tests = ();
71 my @categories = ();
72 my %uniqueCategories = ();
73
74 sub loadTestsList()
75 {
76     open TESTLIST, "<", "tests/LIST" or die;
77     while (<TESTLIST>) {
78         chomp;
79         next unless !$testsPattern || /$testsPattern/;
80         
81         push @tests, $_;
82         my $category = $_;
83         $category =~ s/-.*//;
84         if (!$uniqueCategories{$category}) {
85             push @categories, $category;
86             $uniqueCategories{$category} = $category;
87         }
88     }
89     close TESTLIST;
90 }
91
92 my $timeString = strftime "%Y-%m-%d-%H.%M.%S", localtime $^T;
93 my $prefixFile = "tmp/sunspider-test-prefix.js";
94 my $resultsFile = "tmp/sunspider-results-$timeString.js";
95
96 sub writePrefixFile()
97 {
98     my $prefix = "var tests = [ " . join(", ", map { '"' . $_ . '"' } @tests) . " ];\n";
99     $prefix .= "var categories = [ " . join(", ", map { '"' . $_ . '"' } @categories) . " ];\n";
100
101     mkdir "tmp";
102     dumpToFile($prefix, $prefixFile);
103 }
104
105 sub runTestsOnce($)
106 {
107     my ($useShark) = @_;
108     my $shellArgs = "-f $prefixFile -f resources/sunspider-standalone-driver.js 2> /dev/null";
109     my $output;
110     if ($useShark) {
111         $output = `shark -i -1 -q "$jsShellPath" $shellArgs`;
112     } else {
113         $output = `"$jsShellPath" $shellArgs | grep -v break`;
114     }
115     return $output;
116 }
117
118 loadTestsList();
119 if ($testsPattern) {
120     print STDERR "Found " . scalar(@tests) . " tests matching '" . $testsPattern . "'\n";
121 } else {
122     print STDERR "Found " . scalar(@tests) . " tests\n";
123 }
124 die "No tests to run"  unless scalar(@tests);
125 print STDERR "Running SunSpider once for warmup, then " . ($runShark ? "under Shark" : "$testRuns time" . ($testRuns == 1 ? "" : "s")) . "\n";
126 writePrefixFile();
127
128 runTestsOnce(0);
129 print "Discarded first run.\n";
130
131 my $result;
132 my $count = 0;
133 my @results = ();
134 my $total = 0;
135 print "[";
136 while ($count++ < $testRuns) {
137     $result = runTestsOnce($runShark);
138     $result =~ s/\r\n/\n/g;
139     chomp $result;
140     push @results, $result;
141     print $result;
142     print ",\n" unless ($count == $testRuns);
143 }
144 print "]\n";
145
146 my $output = "var output = [\n" . join(",\n", @results) . "\n];\n";
147 dumpToFile($output, $resultsFile);
148
149 system("$jsShellPath", "-f", $prefixFile, "-f", $resultsFile, "-f", "resources/sunspider-analyze-results.js");
150
151 if ($runShark) {
152     my $newestAge = 0;
153     my $newestMShark = 0;
154     opendir DIR, "." or die;
155     for my $file (readdir DIR) {
156         if ($file =~ /\.mshark$/) {
157             my $age = -M $file;
158             if ($age < $newestAge) {
159                 $newestMShark = $file;
160                 $newestAge = $age;
161             }
162         }
163     }
164     closedir DIR;
165     if ($newestMShark) {
166         my $profileFile = "tmp/sunspider-profile-$timeString.mshark";
167         rename $newestMShark, $profileFile or die;
168         exec "/usr/bin/open", $profileFile;
169     }
170 }