Rename WebKitTools to Tools
[WebKit-https.git] / Tools / Scripts / build-webkit
1 #!/usr/bin/perl -w
2
3 # Copyright (C) 2005, 2006, 2007, 2008, 2009, 2010 Apple Inc. All rights reserved.
4 # Copyright (C) 2009 Google Inc. All rights reserved.
5 # Copyright (C) 2010 moiji-mobile.com All rights reserved.
6 #
7 # Redistribution and use in source and binary forms, with or without
8 # modification, are permitted provided that the following conditions
9 # are met:
10 #
11 # 1.  Redistributions of source code must retain the above copyright
12 #     notice, this list of conditions and the following disclaimer. 
13 # 2.  Redistributions in binary form must reproduce the above copyright
14 #     notice, this list of conditions and the following disclaimer in the
15 #     documentation and/or other materials provided with the distribution. 
16 # 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
17 #     its contributors may be used to endorse or promote products derived
18 #     from this software without specific prior written permission. 
19 #
20 # THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
21 # EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
22 # WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
23 # DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
24 # DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
25 # (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
26 # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
27 # ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
29 # THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
31 # Build script wrapper for the WebKit Open Source Project.
32
33 use strict;
34 use File::Basename;
35 use File::Find;
36 use File::Spec;
37 use FindBin;
38 use Getopt::Long qw(:config pass_through);
39 use lib $FindBin::Bin;
40 use webkitdirs;
41 use webkitperl::features;
42 use POSIX;
43
44 sub formatBuildTime($);
45 sub writeCongrats();
46
47 my $originalWorkingDirectory = getcwd();
48 chdirWebKit();
49
50 my $showHelp = 0;
51 my $clean = 0;
52 my $minimal = 0;
53 my $v8 = 0;
54 my $installHeaders;
55 my $installLibs;
56 my $prefixPath;
57 my $makeArgs;
58 my $noWebKit2 = 0;
59 my $startTime = time();
60
61 my (
62     $threeDCanvasSupport,
63     $threeDRenderingSupport,
64     $accelerated2dCanvasSupport,
65     $blobSupport,
66     $channelMessagingSupport,
67     $clientBasedGeolocationSupport,
68     $coverageSupport,
69     $databaseSupport,
70     $datagridSupport,
71     $datalistSupport,
72     $deviceOrientationSupport,
73     $directoryUploadSupport,
74     $domStorageSupport,
75     $eventsourceSupport,
76     $fileSystemSupport,
77     $filtersSupport,
78     $fullscreenAPISupport,
79     $geolocationSupport,
80     $iconDatabaseSupport,
81     $imageResizerSupport,
82     $indexedDatabaseSupport,
83     $inputSpeechSupport,
84     $javaScriptDebuggerSupport,
85     $linkPrefetchSupport,
86     $mathmlSupport,
87     $meterTagSupport,
88     $notificationsSupport,
89     $offlineWebApplicationSupport,
90     $progressTagSupport,
91     $sharedWorkersSupport,
92     $svgSupport,
93     $svgAnimationSupport,
94     $svgAsImageSupport,
95     $svgDOMObjCBindingsSupport,
96     $svgFontsSupport,
97     $svgForeignObjectSupport,
98     $svgUseSupport,
99     $systemMallocSupport,
100     $tiledBackingStoreSupport,
101     $videoSupport,
102     $wcssSupport,
103     $webAudioSupport,
104     $webInspectorSupport,
105     $webSocketsSupport,
106     $webTimingSupport,
107     $wmlSupport,
108     $workersSupport,
109     $xhtmlmpSupport,
110     $xpathSupport,
111     $xsltSupport,
112 );
113
114 my @features = (
115     { option => "3d-canvas", desc => "Toggle 3D canvas support",
116       define => "ENABLE_3D_CANVAS", default => (isAppleMacWebKit() && !isTiger() && !isLeopard()), value => \$threeDCanvasSupport },
117
118     { option => "3d-rendering", desc => "Toggle 3D rendering support",
119       define => "ENABLE_3D_RENDERING", default => (isAppleMacWebKit() && !isTiger()), value => \$threeDRenderingSupport },
120
121     { option => "accelerated-2d-canvas", desc => "Toggle accelerated 2D canvas support",
122       define => "ENABLE_ACCELERATED_2D_CANVAS", default => 0, value => \$accelerated2dCanvasSupport },
123
124     { option => "blob", desc => "Toggle Blob support",
125       define => "ENABLE_BLOB", default => (isAppleMacWebKit()), value => \$blobSupport },
126
127     { option => "channel-messaging", desc => "Toggle MessageChannel and MessagePort support",
128       define => "ENABLE_CHANNEL_MESSAGING", default => 1, value => \$channelMessagingSupport },
129
130     { option => "client-based-geolocation", desc => "Toggle client-based Geolocation support",
131       define => "ENABLE_CLIENT_BASED_GEOLOCATION", default => isAppleWebKit(), value => \$clientBasedGeolocationSupport },
132
133     { option => "coverage", desc => "Toggle code coverage support",
134       define => "", default => 0, value => \$coverageSupport },
135
136     { option => "database", desc => "Toggle Database Support",
137       define => "ENABLE_DATABASE", default => 1, value => \$databaseSupport },
138
139     { option => "datagrid", desc => "Toggle Datagrid Support",
140       define => "ENABLE_DATAGRID", default => 0, value => \$datagridSupport },
141
142     { option => "datalist", desc => "Toggle HTML5 datalist support",
143       define => "ENABLE_DATALIST", default => 1, value => \$datalistSupport },
144
145     { option => "device-orientation", desc => "Toggle DeviceOrientation support",
146       define => "ENABLE_DEVICE_ORIENTATION", default => 0, value => \$deviceOrientationSupport },
147
148     { option => "directory-upload", desc => "Toogle Directory upload support",
149       define => "ENABLE_DIRECTORY_UPLOAD", default => 0, value => \$directoryUploadSupport },
150
151     { option => "dom-storage", desc => "Toggle DOM Storage Support",
152       define => "ENABLE_DOM_STORAGE", default => 1, value => \$domStorageSupport },
153
154     { option => "eventsource", desc => "Toggle server-sent events support",
155       define => "ENABLE_EVENTSOURCE", default => 1, value => \$eventsourceSupport },
156
157     { option => "file-system", desc => "Toggle FileSystem support",
158       define => "ENABLE_FILE_SYSTEM", default => 0, value => \$fileSystemSupport },
159
160     { option => "filters", desc => "Toggle Filters support",
161       define => "ENABLE_FILTERS", default => (isAppleWebKit() || isGtk() || isQt() || isEfl()), value => \$filtersSupport },
162
163     { option => "fullscreen-api", desc => "Toggle Fullscreen API support",
164       define => "ENABLE_FULLSCREEN_API", default => (isAppleMacWebKit() || isGtk()), value => \$fullscreenAPISupport },
165
166     { option => "geolocation", desc => "Toggle Geolocation support",
167       define => "ENABLE_GEOLOCATION", default => (isAppleWebKit() || isGtk()), value => \$geolocationSupport },
168
169     { option => "icon-database", desc => "Toggle Icon database support",
170       define => "ENABLE_ICONDATABASE", default => 1, value => \$iconDatabaseSupport },
171
172     { option => "image-resizer", desc => "Toggle Image Resizer API support",
173       define => "ENABLE_IMAGE_RESIZER", default => 0, value => \$imageResizerSupport },
174
175     { option => "indexed-database", desc => "Toggle Indexed Database API support",
176       define => "ENABLE_INDEXED_DATABASE", default => 0, value => \$indexedDatabaseSupport },
177
178     { option => "input-speech", desc => "Speech Input API support",
179       define => "ENABLE_INPUT_SPEECH", default => 0, value => \$inputSpeechSupport },
180
181     { option => "inspector", desc => "Toggle Web Inspector support",
182       define => "ENABLE_INSPECTOR", default => 1, value => \$webInspectorSupport },
183
184     { option => "javascript-debugger", desc => "Toggle JavaScript Debugger/Profiler support",
185       define => "ENABLE_JAVASCRIPT_DEBUGGER", default => 1, value => \$javaScriptDebuggerSupport },
186
187     { option => "link-prefetch", desc => "Toggle pre fetching support",
188       define => "ENABLE_LINK_PREFETCH", default => 0, value => \$linkPrefetchSupport },
189
190     { option => "mathml", desc => "Toggle MathML support",
191       define => "ENABLE_MATHML", default => 1, value => \$mathmlSupport },
192
193     { option => "meter-tag", desc => "Meter Tag support",
194       define => "ENABLE_METER_TAG", default => !isGtk() && !isAppleWinWebKit(), value => \$meterTagSupport },
195
196     { option => "notifications", desc => "Toggle Desktop Notifications Support",
197       define => "ENABLE_NOTIFICATIONS", default => 0, value => \$notificationsSupport },
198
199     { option => "offline-web-applications", desc => "Toggle Offline Web Application Support",
200       define => "ENABLE_OFFLINE_WEB_APPLICATIONS", default => 1, value => \$offlineWebApplicationSupport },
201
202     { option => "progress-tag", desc => "Progress Tag support",
203       define => "ENABLE_PROGRESS_TAG", default => 1, value => \$progressTagSupport },
204
205     { option => "system-malloc", desc => "Toggle system allocator instead of TCmalloc",
206       define => "USE_SYSTEM_MALLOC", default => 0, value => \$systemMallocSupport },
207
208     { option => "shared-workers", desc => "Toggle SharedWorkers support",
209       define => "ENABLE_SHARED_WORKERS", default => (isAppleWebKit() || isGtk()), value => \$sharedWorkersSupport },
210
211     { option => "svg", desc => "Toggle SVG support",
212       define => "ENABLE_SVG", default => 1, value => \$svgSupport },
213
214     { option => "svg-animation", desc => "Toggle SVG animation support (implies SVG support)",
215       define => "ENABLE_SVG_ANIMATION", default => 1, value => \$svgAnimationSupport },
216
217     { option => "svg-as-image", desc => "Toggle SVG as Image support (implies SVG support)",
218       define => "ENABLE_SVG_AS_IMAGE", default => 1, value => \$svgAsImageSupport },
219
220     { option => "svg-dom-objc-bindings", desc => "Toggle SVG DOM Objective-C bindings support (implies SVG support)",
221       define => "ENABLE_SVG_DOM_OBJC_BINDINGS", default => isAppleMacWebKit(), value => \$svgDOMObjCBindingsSupport },
222
223     { option => "svg-fonts", desc => "Toggle SVG fonts support (imples SVG support)",
224       define => "ENABLE_SVG_FONTS", default => 1, value => \$svgFontsSupport },
225
226     { option => "svg-foreign-object", desc => "Toggle SVG foreign object support (implies SVG support)",
227       define => "ENABLE_SVG_FOREIGN_OBJECT", default => 1, value => \$svgForeignObjectSupport },
228
229     { option => "svg-use", desc => "Toggle SVG use element support (implies SVG support)",
230       define => "ENABLE_SVG_USE", default => 1, value => \$svgUseSupport },
231
232     { option => "tiled-backing-store", desc => "Toggle Tiled Backing Store support",
233       define => "ENABLE_TILED_BACKING_STORE", default => isQt(), value => \$tiledBackingStoreSupport },
234
235     { option => "video", desc => "Toggle Video support",
236       define => "ENABLE_VIDEO", default => (isAppleWebKit() || isGtk()), value => \$videoSupport },
237
238     { option => "wcss", desc => "Toggle WCSS support",
239       define => "ENABLE_WCSS", default => 0, value => \$wcssSupport },
240
241     { option => "web-audio", desc => "Toggle Web Audio support",
242       define => "ENABLE_WEB_AUDIO", default => 0, value=> \$webAudioSupport },
243
244     { option => "web-sockets", desc => "Toggle Web Sockets support",
245       define => "ENABLE_WEB_SOCKETS", default => 1, value=> \$webSocketsSupport },
246
247     { option => "web-timing", desc => "Toggle Web Timing support",
248       define => "ENABLE_WEB_TIMING", default => 0, value=> \$webTimingSupport },
249
250     { option => "wml", desc => "Toggle WML support",
251       define => "ENABLE_WML", default => 0, value => \$wmlSupport },
252
253     { option => "workers", desc => "Toggle Web Workers support",
254       define => "ENABLE_WORKERS", default => (isAppleWebKit() || isGtk()), value => \$workersSupport },
255
256     { option => "xhtmlmp", desc => "Toggle XHTML-MP support",
257       define => "ENABLE_XHTMLMP", default => 0, value => \$xhtmlmpSupport },
258
259     { option => "xpath", desc => "Toggle XPath support",
260       define => "ENABLE_XPATH", default => 1, value => \$xpathSupport },
261
262     { option => "xslt", desc => "Toggle XSLT support",
263       define => "ENABLE_XSLT", default => 1, value => \$xsltSupport },
264 );
265
266 # Update defaults from Qt's project file
267 if (isQt()) {
268     my %qtDefaults = qtFeatureDefaults();
269     foreach (@features) {
270         $_->{default} = $qtDefaults{$_->{define}} || 0;
271     }
272 }
273
274 # Additional environment parameters
275 push @ARGV, split(/ /, $ENV{'BUILD_WEBKIT_ARGS'}) if ($ENV{'BUILD_WEBKIT_ARGS'});
276
277 # Initialize values from defaults
278 foreach (@ARGV) {
279     if ($_ eq '--minimal') {
280         $minimal = 1;
281     } elsif ($_ eq '--v8') {
282         $v8 = 1;
283     }
284 }
285
286 # Initialize values from defaults
287 foreach (@features) {
288     ${$_->{value}} = ($_->{default} && !$minimal) || 0;
289 }
290
291 $svgSupport = $svgSupport || $svgAnimationSupport || $svgAsImageSupport
292     || $svgDOMObjCBindingsSupport || $svgFontsSupport
293     || $svgForeignObjectSupport || $svgUseSupport;
294
295
296 my $programName = basename($0);
297 my $usage = <<EOF;
298 Usage: $programName [options] [options to pass to build system]
299   --help                            Show this help message
300   --clean                           Cleanup the build directory
301   --debug                           Compile in debug mode
302   --wincairo                        Build using Cairo (rather than CoreGraphics) on Windows
303   --chromium                        Build the Chromium port on Mac/Win/Linux
304   --gtk                             Build the GTK+ port
305   --qt                              Build the Qt port
306   --efl                             Build the EFL port
307   --inspector-frontend              Copy changes to the inspector front-end files to the build directory
308
309   --install-headers=<path>          Set installation path for the headers (Qt only)
310   --install-libs=<path>             Set installation path for the libraries (Qt only)
311   --v8                              Use V8 as JavaScript engine (Qt only)
312
313   --prefix=<path>                   Set installation prefix to the given path (Gtk/Efl only)
314   --makeargs=<arguments>            Optional Makefile flags
315
316   --minimal                         No optional features, unless explicitly enabled.
317
318 EOF
319
320 my %options = (
321     'help' => \$showHelp,
322     'clean' => \$clean,
323     'install-headers=s' => \$installHeaders,
324     'install-libs=s' => \$installLibs,
325     'prefix=s' => \$prefixPath,
326     'makeargs=s' => \$makeArgs,
327     'minimal' => \$minimal,
328     'v8' => \$v8,
329     'no-webkit2' => \$noWebKit2,
330 );
331
332 # Build usage text and options list from features
333 foreach (@features) {
334     my $opt = sprintf("%-35s", "  --[no-]$_->{option}");
335     $usage .= "$opt $_->{desc} (default: $_->{default})\n";
336     $options{"$_->{option}!"} = $_->{value};
337 }
338
339 GetOptions(%options);
340
341 if ($showHelp) {
342    print STDERR $usage;
343    exit 1;
344 }
345
346 checkRequiredSystemConfig();
347 setConfiguration();
348
349 my $productDir = productDir();
350
351 # Remove 0 byte sized files from productDir after slave lost for Qt buildbots.
352 File::Find::find(\&unlinkZeroFiles, $productDir) if (isQt() && -e $productDir);
353
354 sub unlinkZeroFiles ()
355 {
356     my $file = $File::Find::name;
357     if (! -s $file) {
358         unlink $file;
359         print "0 byte sized file removed from build directory: $file\n";
360     }
361 }
362
363 # Check that all the project directories are there.
364 my @projects = ("JavaScriptCore", "WebCore", "WebKit");
365
366 my @otherDirs = ("WebKitLibraries");
367 for my $dir (@projects, @otherDirs) {
368     if (! -d $dir) {
369         die "Error: No $dir directory found. Please do a fresh checkout.\n";
370     }
371 }
372
373 my @options = ();
374
375 # enable autotool options accordingly
376 if (isGtk()) {
377     @options = @ARGV;
378     foreach (@features) {
379         push @options, autotoolsFlag(${$_->{value}}, $_->{option});
380     }
381
382     push @options, "--prefix=" . $prefixPath if defined($prefixPath);
383     push @options, "--makeargs=" . $makeArgs if defined($makeArgs);
384 } elsif (isAppleMacWebKit()) {
385     push @options, XcodeOptions();
386
387     sub option($$$)
388     {
389         my ($feature, $isEnabled, $defaultValue) = @_;
390         return "" if $defaultValue == $isEnabled;
391         return $feature . "=" . ($isEnabled ? $feature : " ");
392     }
393
394     foreach (@features) {
395         if ($_->{option} ne "coverage") {
396             my $option = option($_->{define}, ${$_->{value}}, $_->{default});
397             push @options, $option unless $option eq "";
398         }
399     }
400
401     # Apple builds JavaScriptGlue, and only on the Mac.
402     splice @projects, 1, 0, "JavaScriptGlue";
403
404     # ANGLE must come before WebCore
405     splice @projects, 0, 0, "ANGLE";
406
407     # WebKit2 is only supported in SnowLeopard and later at present.
408     push @projects, ("WebKit2", "Tools/MiniBrowser") if osXVersion()->{"minor"} >= 6 and !$noWebKit2;
409
410     # Copy library and header from WebKitLibraries to a findable place in the product directory.
411     my @librariesToCopy = (
412         "libWebKitSystemInterfaceTiger.a",
413         "libWebKitSystemInterfaceLeopard.a",
414         "libWebKitSystemInterfaceSnowLeopard.a",
415         "libWebCoreSQLite3.a",
416     );
417     foreach my $libName (@librariesToCopy) {
418         my $srcLib = "WebKitLibraries/" . $libName;
419         my $lib = "$productDir/" . $libName;
420         if (!-e $lib || -M $lib > -M $srcLib) {
421             print "Updating $lib\n";
422             system "ditto", $srcLib, $lib;
423             system "ranlib", $lib;
424         }
425     }
426
427     # FIXME: This code should be abstracted to not be copy/paste.
428     my $srcHeader = "WebKitLibraries/WebKitSystemInterface.h";
429     my $header = "$productDir/usr/local/include/WebKitSystemInterface.h";
430     if (!-e $header || -M $header > -M $srcHeader) {
431         print "Updating $header\n";
432         system "mkdir", "-p", "$productDir/usr/local/include";
433         system "ditto", $srcHeader, $header;
434     }
435
436     my $srcHeaderDir = "WebKitLibraries/WebCoreSQLite3";
437     my $headerDir = "$productDir/WebCoreSQLite3";
438     if (!-e $headerDir || -M $headerDir > -M $srcHeaderDir) {
439         print "Updating $headerDir\n";
440         system "ditto", $srcHeaderDir, $headerDir;
441     }
442 } elsif (isAppleWinWebKit()) {
443     # Copy WebKitSupportLibrary to the correct location in WebKitLibraries so it can be found.
444     # Will fail if WebKitSupportLibrary.zip is not in source root.
445     (system("perl Tools/Scripts/update-webkit-support-libs") == 0) or die;
446 } elsif (isQt()) {
447     @options = @ARGV;
448     push @options, "--install-headers=" . $installHeaders if defined($installHeaders);
449     push @options, "--install-libs=" . $installLibs if defined($installLibs);
450     push @options, "--makeargs=" . $makeArgs if defined($makeArgs);
451
452     foreach (@features) {
453         push @options, "DEFINES+=$_->{define}=${$_->{value}}" if ${$_->{value}} != $_->{default};
454     }
455
456     if ($minimal) {
457         push @options, "CONFIG+=minimal";
458     }
459
460     if ($v8) {
461         push @options, "CONFIG+=v8";
462     }
463 }
464
465 # Force re-link of existing libraries if different than expected
466 removeLibraryDependingOnFeature("WebCore", "SVG", $svgSupport);
467
468 if (isInspectorFrontend()) {
469     exit exitStatus(copyInspectorFrontendFiles());
470 }
471
472 if (isWx()) {
473     downloadWafIfNeeded();
474     @options = ();
475     if (defined($makeArgs)) {
476         @options = split(/ /, $makeArgs);
477     }
478     @projects = ();
479     my $result = buildWafProject('.', $clean, @options);
480     exit exitStatus($result) if exitStatus($result);
481 }
482
483 if (isChromium()) {
484     @options = @ARGV;
485     # Chromium doesn't build by project directories.
486     @projects = ();
487     my $result = buildChromium($clean, @options);
488     exit exitStatus($result) if exitStatus($result);
489 }
490
491 if (isEfl()) {
492     @options = ();
493     @projects = ();
494     foreach (@features) {
495         my $featureName = $_->{define};
496         if ($featureName) {
497             my $featureEnabled = ${$_->{value}} ? "ON" : "OFF";
498             push @options, "-D$featureName=$featureEnabled";
499         }
500     }
501     push @options, "--makeargs=" . $makeArgs if defined($makeArgs);
502     push @options, "--prefix=" . $prefixPath if defined($prefixPath);
503     my $result = buildCMakeEflProject($clean, @options);
504     exit exitStatus($result) if exitStatus($result);
505 }
506
507 # Build, and abort if the build fails.
508 for my $dir (@projects) {
509     chdir $dir or die;
510     my $result = 0;
511
512     # For Gtk and Qt the WebKit project builds all others
513     if ((isGtk() || isQt()) && $dir ne "WebKit") {
514         chdir ".." or die;
515         next;
516     }
517
518     if (isGtk()) {
519         $result = buildGtkProject($dir, $clean,  @options);
520     } elsif (isQt()) {
521         $result = buildQMakeQtProject($dir, $clean, @options);
522     } elsif (isAppleMacWebKit()) {
523         $dir = "MiniBrowser" if $dir eq "Tools/MiniBrowser";
524         my @local_options = @options;
525         push @local_options, XcodeCoverageSupportOptions() if $coverageSupport && $dir ne "ANGLE";
526         $result = buildXCodeProject($dir, $clean, @local_options, @ARGV);
527     } elsif (isAppleWinWebKit()) {
528         if ($dir eq "WebKit") {
529             $result = buildVisualStudioProject("win/WebKit.vcproj/WebKit.sln", $clean);
530         }
531     }
532     # Various build* calls above may change the CWD.
533     chdirWebKit();
534
535     if (exitStatus($result)) {
536         my $scriptDir = relativeScriptsDir();
537         if (usingVisualStudioExpress()) {
538             # Visual Studio Express is so lame it can't stdout build failures.
539             # So we find its logs and dump them to the console ourselves.
540             system(File::Spec->catfile($scriptDir, "print-vse-failure-logs"));
541         }
542         if (isAppleWinWebKit()) {
543             print "\n\n===== BUILD FAILED ======\n\n";
544             print "Please ensure you have run $scriptDir/update-webkit to install dependencies.\n\n";
545             my $baseProductDir = baseProductDir();
546             print "You can view build errors by checking the BuildLog.htm files located at:\n$baseProductDir/obj/<project>/<config>.\n";
547         }
548         exit exitStatus($result);
549     }
550 }
551
552 # Don't report the "WebKit is now built" message after a clean operation.
553 exit if $clean;
554
555 # Write out congratulations message.
556 writeCongrats();
557
558 exit 0;
559
560 sub formatBuildTime($)
561 {
562     my ($buildTime) = @_;
563
564     my $buildHours = int($buildTime / 3600);
565     my $buildMins = int(($buildTime - $buildHours * 3600) / 60);
566     my $buildSecs = $buildTime - $buildHours * 3600 - $buildMins * 60;
567
568     if ($buildHours) {
569         return sprintf("%dh:%02dm:%02ds", $buildHours, $buildMins, $buildSecs);
570     }
571     return sprintf("%02dm:%02ds", $buildMins, $buildSecs);
572 }
573
574 sub writeCongrats()
575 {
576     my $launcherPath = launcherPath();
577     my $launcherName = launcherName();
578     my $endTime = time();
579     my $buildTime = formatBuildTime($endTime - $startTime);
580
581     print "\n";
582     print "===========================================================\n";
583     print " WebKit is now built ($buildTime). \n";
584     if (!isChromium()) {
585         print " To run $launcherName with this newly-built code, use the\n";
586         print " \"$launcherPath\" script.\n";
587     }
588     print "===========================================================\n";
589 }