2 # ***** BEGIN LICENSE BLOCK *****
3 # Version: MPL 1.1/GPL 2.0/LGPL 2.1
5 # The contents of this file are subject to the Mozilla Public License Version
6 # 1.1 (the "License"); you may not use this file except in compliance with
7 # the License. You may obtain a copy of the License at
8 # http://www.mozilla.org/MPL/
10 # Software distributed under the License is distributed on an "AS IS" basis,
11 # WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
12 # for the specific language governing rights and limitations under the
15 # The Original Code is the Mozilla Mac OS X Universal Binary Packaging System
17 # The Initial Developer of the Original Code is Google Inc.
18 # Portions created by the Initial Developer are Copyright (C) 2006
19 # the Initial Developer. All Rights Reserved.
22 # Mark Mentovai <mark@moxienet.com> (Original Author)
24 # Alternatively, the contents of this file may be used under the terms of
25 # either the GNU General Public License Version 2 or later (the "GPL"), or
26 # the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
27 # in which case the provisions of the GPL or the LGPL are applicable instead
28 # of those above. If you wish to allow use of your version of this file only
29 # under the terms of either the GPL or the LGPL, and not to allow others to
30 # use your version of this file under the terms of the MPL, indicate your
31 # decision by deleting the provisions above and replace them with the notice
32 # and other provisions required by the GPL or the LGPL. If you do not delete
33 # the provisions above, a recipient may use your version of this file under
34 # the terms of any one of the MPL, the GPL or the LGPL.
36 # ***** END LICENSE BLOCK *****
45 B<unify> - Mac OS X universal binary packager
54 [B<--only-one> I<action>]
55 [B<--verbosity> I<level>]
59 I<unify> merges any two architecture-specific files or directory trees
60 into a single file or tree suitable for use on either architecture as a
61 "fat" or "universal binary."
63 Architecture-specific Mach-O files will be merged into fat Mach-O files
64 using L<lipo(1)>. Non-Mach-O files in the architecture-specific trees
65 are compared to ensure that they are equivalent before copying. Symbolic
66 links are permitted in the architecture-specific trees and will cause
67 identical links to be created in the merged tree, provided that the source
68 links have identical targets. Directories are processed recursively.
70 If the architecture-specific source trees contain zip archives (including
71 jar files) that are not identical according to a byte-for-byte check, they
72 are still assumed to be equivalent if both archives contain exactly the
73 same members with identical checksums and sizes.
75 Behavior when one architecture-specific tree contains files that the other
76 does not is controlled by the B<--only-one> option.
78 If Mach-O files cannot be merged using L<lipo(1)>, zip archives are not
79 equivalent, regular files are not identical, or any other error occurs,
80 B<unify> will fail with an exit status of 1. Diagnostic messages are
81 typically printed to stderr; this behavior can be controlled with the
82 B<--verbosity> option.
92 The paths to directory trees containing PowerPC and x86 builds,
93 respectively. I<ppc-path> and I<x86-path> are permitted to contain files
94 that are already "fat," and only the appropriate architecture's images will
97 I<ppc-path> and I<x86-path> are also permitted to both be files, in which
98 case B<unify> operates solely on those files, and produces an appropriate
99 merged file at I<target-path>.
103 The path to the merged file or directory tree. This path will be created,
104 and it must not exist prior to running B<unify>.
108 When specified, the commands that would be executed are printed, without
109 actually executing them. Note that B<--dry-run> and the equivalent
110 B<--verbosity> level during "wet" runs may print equivalent commands when
111 no commands are in fact executed: certain operations are handled internally
112 within B<unify>, and an approximation of a command that performs a similar
115 =item B<--only-one> I<action>
117 Controls handling of files that are only present in one of the two source
118 trees. I<action> may be:
119 skip - These files are skipped.
120 copy - These files are copied from the tree in which they exist.
121 fail - When this condition occurs, it is treated as an error.
123 The default I<action> is copy.
125 =item B<--verbosity> I<level>
127 Adjusts the level of loudness of B<unify>. The possible values for
129 0 - B<unify> never prints anything.
130 (Other programs that B<unify> calls may still print messages.)
131 1 - Fatal error messages are printed to stderr.
132 2 - Nonfatal warnings are printed to stderr.
133 3 - Commands are printed to stdout as they are executed.
135 The default I<level> is 2.
143 =item Create a universal .app bundle from two architecture-specific .app
146 unify --only-one copy ppc/dist/firefox/Firefox.app
147 x86/dist/firefox/Firefox.app universal/Firefox.app
150 =item Merge two identical architecture-specific trees:
152 unify --only-one fail /usr/local /nfs/x86/usr/local
159 The only esoteric requirement of B<unify> is that the L<lipo(1)> command
160 be available. It is present on Mac OS X systems at least as early as
161 10.3.9, and probably earlier. Mac OS X 10.4 ("Tiger") or later are
166 MPL 1.1/GPL 2.0/LGPL 2.1. Your choice
170 The software was initially written by Mark Mentovai; copyright 2006
175 L<cmp(1)>, L<ditto(1)>, L<lipo(1)>
179 use Archive
::Zip
(':ERROR_CODES');
186 my (%gConfig, $gDryRun, $gOnlyOne, $gVerbosity);
188 sub argumentEscape
(@
);
190 sub compareZipArchives
($$);
192 sub copyIfIdentical
($$$);
193 sub createUniqueFile
($$);
194 sub makeUniversal
($$$);
195 sub makeUniversalDirectory
($$$);
196 sub makeUniversalInternal
($$$$);
197 sub makeUniversalFile
($$$);
202 package FileAttrCache
;
210 sub lIsExecutable
($);
211 sub lIsRegularFile
($);
223 'cmd_lipo' => 'lipo',
231 Getopt
::Long
::Configure
('pass_through');
232 GetOptions
('dry-run' => \
$gDryRun,
233 'only-one=s' => \
$gOnlyOne,
234 'verbosity=i' => \
$gVerbosity,
235 'config=s' => \
%gConfig); # "hidden" option not in usage()
237 if (scalar(@ARGV) != 3 || $gVerbosity < 0 || $gVerbosity > 3 ||
238 ($gOnlyOne ne 'skip' && $gOnlyOne ne 'copy' && $gOnlyOne ne 'fail')) {
243 if (!makeUniversal
($ARGV[0],$ARGV[1],$ARGV[2])) {
244 # makeUniversal or something it called will have printed an error.
250 # argumentEscape(@arguments)
252 # Takes a list of @arguments and makes them shell-safe.
253 sub argumentEscape
(@
) {
257 my ($argument, @argumentsOut);
258 foreach $argument (@arguments) {
259 $argument =~ s
%([^A
-Za
-z0
-9_\
-/.=+,])%\\$1%g;
260 push(@argumentsOut, $argument);
263 return @argumentsOut;
266 # command(@arguments)
268 # Runs the specified command by calling system(@arguments). If $gDryRun
269 # is true, the command is printed but not executed, and 0 is returned.
270 # if $gVerbosity is greater than 1, the command is printed before being
271 # executed. When the command is executed, the system() return value will
272 # be returned. stdout and stderr are left connected for command output.
276 if ($gVerbosity >= 3 || $gDryRun) {
277 print(join(' ', argumentEscape
(@arguments))."\n");
282 return system(@arguments);
285 # compareZipArchives($zip1, $zip2)
287 # Given two pathnames to zip archives, determines whether or not they are
288 # functionally identical. Returns true if they are, false if they differ in
289 # some substantial way, and undef if an error occurs. If the zip files
290 # differ, diagnostic messages are printed indicating how they differ.
292 # Zip files will differ if any of the members are different as defined by
293 # readZipCRCs, which consider CRCs, sizes, and file types as stored in the
294 # file header. Timestamps are not considered. Zip files also differ if one
295 # file contains members that the other one does not. $gOnlyOne has no
296 # effect on this behavior.
297 sub compareZipArchives
($$) {
301 my ($CRCHash1, $CRCHash2);
302 if (!defined($CRCHash1 = readZipCRCs
($zip1))) {
303 # readZipCRCs printed an error.
306 if (!defined($CRCHash2 = readZipCRCs
($zip2))) {
307 # readZipCRCs printed an error.
311 my (@diffCRCs, @onlyInZip1);
316 foreach $memberName (keys(%$CRCHash1)) {
317 if (!exists($$CRCHash2{$memberName})) {
318 # The member is present in $zip1 but not $zip2.
319 push(@onlyInZip1, $memberName);
321 elsif ($$CRCHash1{$memberName} ne $$CRCHash2{$memberName}) {
322 # The member is present in both archives but its CRC or some other
323 # other critical attribute isn't identical.
324 push(@diffCRCs, $memberName);
326 delete($$CRCHash2{$memberName});
329 # If any members remain in %CRCHash2, it's because they're not present
332 @onlyInZip2 = keys(%$CRCHash2);
334 if (scalar(@onlyInZip1) + scalar(@onlyInZip2) + scalar(@diffCRCs)) {
335 complain
(1, 'compareZipArchives: zip archives differ:',
338 if (scalar(@onlyInZip1)) {
339 complain
(1, 'compareZipArchives: members only in former:',
342 if (scalar(@onlyInZip2)) {
343 complain
(1, 'compareZipArchives: members only in latter:',
346 if (scalar(@diffCRCs)) {
347 complain
(1, 'compareZipArchives: members differ:',
356 # complain($severity, $message, @list)
358 # Prints $message to stderr if $gVerbosity allows it for severity level
359 # $severity. @list is a list of words that will be shell-escaped and printed
360 # after $message, one per line, intended to be used, for example, to list
361 # arguments to a call that failed.
363 # Expected severity levels are 1 for hard errors and 2 for non-fatal warnings.
365 # Always returns false as a convenience, so callers can return complain's
366 # return value when it is used to signal errors.
368 my ($severity, $message, @list);
369 ($severity, $message, @list) = @_;
371 if ($gVerbosity >= $severity) {
372 print STDERR
($0.': '.$message."\n");
375 while ($item = shift(@list)) {
376 print STDERR
(' '.(argumentEscape
($item))[0].
377 (scalar(@list)?
',':'')."\n");
384 # copyIfIdentical($source1, $source2, $target)
386 # $source1 and $source2 are FileAttrCache objects that are compared, and if
387 # identical, copied to path string $target. The comparison is initially
388 # done as a byte-for-byte comparison, but if the files differ and appear to
389 # be zip archives, compareZipArchives is called to determine whether
390 # files that are not byte-for-byte identical are equivalent archives.
392 # Returns true on success, false for files that are not identical or
393 # equivalent archives, and undef if an error occurs.
395 # One of $source1 and $source2 is permitted to be undef. In this event,
396 # whichever source is defined is copied directly to $target without performing
397 # any comparisons. This enables the $gOnlyOne = 'copy' mode, which is
398 # driven by makeUniversalDirectory and makeUniversalInternal.
399 sub copyIfIdentical
($$$) {
400 my ($source1, $source2, $target);
401 ($source1, $source2, $target) = @_;
403 if (!defined($source1)) {
404 # If there's only one source file, make it the first file. Order
405 # isn't important here, and this makes it possible to use
406 # defined($source2) as the switch, and to always copy from $source1.
411 if (defined($source2)) {
412 # Only do the comparisons if there are two source files. If there's
413 # only one source file, skip the comparisons and go straight to the
415 if ($gVerbosity >= 3 || $gDryRun) {
417 join(' ',argumentEscape
($source1->path(), $source2->path()))."\n");
420 if (!defined($comparison = compare
($source1->path(), $source2->path())) ||
422 return complain
(1, 'copyIfIdentical: compare: '.$!.' while comparing:',
426 elsif ($comparison != 0) {
428 if (defined($zip1 = $source1->isZip()) &&
429 defined($zip2 = $source2->isZip()) &&
432 if (!defined($zipComparison = compareZipArchives
($source1->path(),
435 # An error occurred or the zip files aren't sufficiently identical.
436 # compareZipArchives will have printed an error message.
439 # The zip files were compared successfully, and they both contain
440 # all of the same members, and all of their members' CRCs are
441 # identical. For the purposes of this script, the zip files can be
442 # treated as identical, so reset $comparison.
446 if ($comparison != 0) {
447 return complain
(1, 'copyIfIdentical: files differ:',
453 if ($gVerbosity >= 3 || $gDryRun) {
455 join(' ',argumentEscape
($source1->path(), $target))."\n");
461 # Set the execute bits (as allowed by the umask) on the new file if any
462 # execute bit is set on either old file.
463 $isExecutable = $source1->lIsExecutable() ||
464 (defined($source2) && $source2->lIsExecutable());
466 if (!createUniqueFile
($target, $isExecutable ?
0777 : 0666)) {
467 # createUniqueFile printed an error.
471 if (!copy
($source1->path(), $target)) {
472 complain
(1, 'copyIfIdentical: copy: '.$!.' while copying',
483 # createUniqueFile($path, $mode)
485 # Creates a new plain empty file at pathname $path, provided it does not
486 # yet exist. $mode is used as the file mode. The actual file's mode will
487 # be modified by the effective umask. Returns false if the file could
488 # not be created, setting $! to the error. An error message is printed
489 # in the event of failure.
490 sub createUniqueFile
($$) {
495 if (!sysopen($fh, $path, O_WRONLY
| O_CREAT
| O_EXCL
, $mode)) {
496 return complain
(1, 'createUniqueFile: open: '.$!.' for:',
504 # makeUniversal($pathPPC, $pathX86, $pathTarget)
506 # The top-level call. $pathPPC, $pathX86, and $pathTarget are strings
507 # identifying the ppc and x86 files or directories to merge and the location
508 # to merge them to. Returns false on failure and true on success.
509 sub makeUniversal
($$$) {
510 my ($pathTarget, $pathPPC, $pathX86);
511 ($pathPPC, $pathX86, $pathTarget) = @_;
513 my ($filePPC, $fileX86);
514 $filePPC = FileAttrCache
->new($pathPPC);
515 $fileX86 = FileAttrCache
->new($pathX86);
517 return makeUniversalInternal
(1, $filePPC, $fileX86, $pathTarget);
520 # makeUniversalDirectory($dirPPC, $dirX86, $dirTarget)
522 # This is part of the heart of recursion. $dirPPC and $dirX86 are
523 # FileAttrCache objects designating the source ppc and x86 directories to
524 # merge into a universal directory at $dirTarget, a string. For each file
525 # in $dirPPC and $dirX86, makeUniversalInternal is called.
526 # makeUniversalInternal will call back into makeUniversalDirectory for
527 # directories, thus completing the recursion. If a failure is encountered
528 # in ths function or in makeUniversalInternal or anything that it calls,
529 # false is returned, otherwise, true is returned.
531 # If there are files present in one source directory but not both, the
532 # value of $gOnlyOne controls the behavior. If $gOnlyOne is 'copy', the
533 # single source file is copied into $pathTarget. If it is 'skip', it is
534 # skipped. If it is 'fail', such files will trigger makeUniversalDirectory
537 # If either source directory is undef, it is treated as having no files.
538 # This facilitates deep recursion when entire directories are only present
539 # in one source when $gOnlyOne = 'copy'.
540 sub makeUniversalDirectory
($$$) {
541 my ($dirPPC, $dirX86, $dirTarget);
542 ($dirPPC, $dirX86, $dirTarget) = @_;
544 my ($dh, @filesPPC, @filesX86);
547 if (defined($dirPPC)) {
548 if (!opendir($dh, $dirPPC->path())) {
549 return complain
(1, 'makeUniversalDirectory: opendir ppc: '.$!.' for:',
552 @filesPPC = readdir($dh);
557 if (defined($dirX86)) {
558 if (!opendir($dh, $dirX86->path())) {
559 return complain
(1, 'makeUniversalDirectory: opendir x86: '.$!.' for:',
562 @filesX86 = readdir($dh);
566 my (%common, $file, %onlyPPC, %onlyX86);
569 foreach $file (@filesPPC) {
570 if ($file eq '.' || $file eq '..') {
578 foreach $file (@filesX86) {
579 if ($file eq '.' || $file eq '..') {
582 if ($onlyPPC{$file}) {
583 delete $onlyPPC{$file};
591 # First, handle files common to both.
592 foreach $file (sort(keys(%common))) {
593 if (!makeUniversalInternal
(0,
594 FileAttrCache
->new($dirPPC->path().'/'.$file),
595 FileAttrCache
->new($dirX86->path().'/'.$file),
596 $dirTarget.'/'.$file)) {
597 # makeUniversalInternal will have printed an error.
602 # Handle files found only in a single directory here. There are three
603 # options, dictated by $gOnlyOne: fail if files are only present in
604 # one directory, skip any files only present in one directory, or copy
605 # these files straight over to the target directory. In any event,
606 # a message will be printed indicating that the file trees don't match
608 if (keys(%onlyPPC)) {
609 complain
(($gOnlyOne eq 'fail' ?
1 : 2),
610 ($gOnlyOne ne 'fail' ?
'warning: ' : '').
611 'makeUniversalDirectory: only in ppc '.
612 (argumentEscape
($dirPPC->path()))[0].':',
613 argumentEscape
(keys(%onlyPPC)));
616 if (keys(%onlyX86)) {
617 complain
(($gOnlyOne eq 'fail' ?
1 : 2),
618 ($gOnlyOne ne 'fail' ?
'warning: ' : '').
619 'makeUniversalDirectory: only in x86 '.
620 (argumentEscape
($dirX86->path()))[0].':',
621 argumentEscape
(keys(%onlyX86)));
624 if ($gOnlyOne eq 'fail' && (keys(%onlyPPC) || keys(%onlyX86))) {
625 # Error message(s) printed above.
629 if ($gOnlyOne eq 'copy') {
630 foreach $file (sort(keys(%onlyPPC))) {
631 if (!makeUniversalInternal
(0,
632 FileAttrCache
->new($dirPPC->path().'/'.$file),
634 $dirTarget.'/'.$file)) {
635 # makeUniversalInternal will have printed an error.
640 foreach $file (sort(keys(%onlyX86))) {
641 if (!makeUniversalInternal
(0,
643 FileAttrCache
->new($dirX86->path().'/'.$file),
644 $dirTarget.'/'.$file)) {
645 # makeUniversalInternal will have printed an error.
654 # makeUniversalFile($sourcePPC, $sourceX86, $targetPath)
656 # Creates a universal file at pathname $targetPath based on a ppc image at
657 # $sourcePPC and an x86 image at $sourceX86. $sourcePPC and $sourceX86 are
658 # both FileAttrCache objects. Returns true on success and false on failure.
659 # On failure, diagnostics will be printed to stderr.
661 # The source files may be either thin Mach-O images of the appropriate
662 # architecture, or fat Mach-O files that contain images of the appropriate
665 # This function wraps the lipo utility, see lipo(1).
666 sub makeUniversalFile
($$$) {
667 my ($sourcePPC, $sourceX86, $targetPath, @tempThinFiles, $thinPPC, $thinX86);
668 ($sourcePPC, $sourceX86, $targetPath) = @_;
669 $thinPPC = $sourcePPC;
670 $thinX86 = $sourceX86;
674 # The source files might already be fat. They should be thinned out to only
675 # contain a single architecture.
677 my ($isFatPPC, $isFatX86);
679 if(!defined($isFatPPC = $sourcePPC->isFat())) {
680 # isFat printed its own error
684 $thinPPC = FileAttrCache
->new($targetPath.'.ppc');
685 push(@tempThinFiles, $thinPPC->path());
686 if (command
($gConfig{'cmd_lipo'}, '-thin', 'ppc',
687 $sourcePPC->path(), '-output', $thinPPC->path()) != 0) {
688 unlink(@tempThinFiles);
689 return complain
(1, 'lipo thin ppc failed for:',
695 if(!defined($isFatX86 = $sourceX86->isFat())) {
696 # isFat printed its own error
697 unlink(@tempThinFiles);
701 $thinX86 = FileAttrCache
->new($targetPath.'.x86');
702 push(@tempThinFiles, $thinX86->path());
703 if (command
($gConfig{'cmd_lipo'}, '-thin', 'i386',
704 $sourceX86->path(), '-output', $thinX86->path()) != 0) {
705 unlink(@tempThinFiles);
706 return complain
(1, 'lipo thin x86 failed for:',
712 # The image for each architecture in the fat file will be aligned on
713 # a specific boundary, default 4096 bytes, see lipo(1) -segalign.
714 # Since there's no tail-padding, the fat file will consume the least
715 # space on disk if the image that comes last exceeds the segment size
716 # by the smallest amount.
718 # This saves an average of 1kB per fat file over the naive approach of
719 # always putting one architecture first: average savings is 2kB per
720 # file, but the naive approach would have gotten it right half of the
723 my ($sizePPC, $sizeX86, $thinPPCForStat, $thinX86ForStat);
726 $thinPPCForStat = $thinPPC;
727 $thinX86ForStat = $thinX86;
730 # Normally, fat source files will have been converted into temporary
731 # thin files. During a dry run, that doesn't happen, so fake it up
732 # a little bit by always using the source file, fat or thin, for the
734 $thinPPCForStat = $sourcePPC;
735 $thinX86ForStat = $sourceX86;
738 if (!defined($sizePPC = $thinPPCForStat->statSize())) {
739 unlink(@tempThinFiles);
740 return complain
(1, 'stat ppc: '.$!.' for:',
741 $thinPPCForStat->path());
743 if (!defined($sizeX86 = $thinX86ForStat->statSize())) {
744 unlink(@tempThinFiles);
745 return complain
(1, 'stat x86: '.$!.' for:',
746 $thinX86ForStat->path());
749 $sizePPC = $sizePPC % 4096;
750 $sizeX86 = $sizeX86 % 4096;
755 # PPC image ends on an alignment boundary, there will be no padding before
756 # starting the x86 image.
757 @thinFiles = ($thinPPC->path(), $thinX86->path());
759 elsif ($sizeX86 == 0 || $sizeX86 > $sizePPC) {
760 # x86 image ends on an alignment boundary, there will be no padding before
761 # starting the PPC image, or the x86 image exceeds its alignment boundary
762 # by more than the PPC image, so there will be less padding if the x86
764 @thinFiles = ($thinX86->path(), $thinPPC->path());
767 # PPC image exceeds its alignment boundary by more than the x86 image, so
768 # there will be less padding if the PPC comes first.
769 @thinFiles = ($thinPPC->path(), $thinX86->path());
773 $isExecutable = $sourcePPC->lIsExecutable() ||
774 $sourceX86->lIsExecutable();
777 # Ensure that the file does not yet exist.
779 # Set the execute bits (as allowed by the umask) on the new file if any
780 # execute bit is set on either old file. Yes, it is possible to have
781 # proper Mach-O files without x-bits: think object files (.o) and static
783 if (!createUniqueFile
($targetPath, $isExecutable ?
0777 : 0666)) {
784 # createUniqueFile printed an error.
785 unlink(@tempThinFiles);
790 # Create the fat file.
791 if (command
($gConfig{'cmd_lipo'}, '-create', @thinFiles,
792 '-output', $targetPath) != 0) {
793 unlink(@tempThinFiles, $targetPath);
794 return complain
(1, 'lipo create fat failed for:',
799 unlink(@tempThinFiles);
802 # lipo seems to think that it's free to set its own file modes that
803 # ignore the umask, which is bogus when the rest of this script
804 # respects the umask.
805 if (!chmod(($isExecutable ?
0777 : 0666) & ~umask(), $targetPath)) {
806 complain
(1, 'makeUniversalFile: chmod: '.$!.' for',
816 # makeUniversalInternal($isToplevel, $filePPC, $fileX86, $fileTargetPath)
818 # Given FileAttrCache objects $filePPC and $fileX86, compares filetypes
819 # and performs the appropriate action to produce a universal file at
820 # path string $fileTargetPath. $isToplevel should be true if this is
821 # the recursive base and false otherwise; this controls cleanup behavior
822 # (cleanup is only performed at the base, because cleanup itself is
825 # This handles regular files by determining whether they are Mach-O files
826 # and calling makeUniversalFile if so and copyIfIdentical otherwise. Symbolic
827 # links are handled directly in this function by ensuring that the source link
828 # targets are identical and creating a new link with the same target
829 # at $fileTargetPath. Directories are handled by calling
830 # makeUniversalDirectory.
832 # One of $filePPC and $fileX86 is permitted to be undef. In that case,
833 # the defined source file is copied directly to the target if a regular
834 # file, and symlinked appropriately if a symbolic link. This facilitates
835 # use of $gOnlyOne = 'copy', although no $gOnlyOne checks are made in this
836 # function, they are all handled in makeUniversalDirectory.
838 # Returns true on success. Returns false on failure, including failures
839 # in other functions called.
840 sub makeUniversalInternal
($$$$) {
841 my ($filePPC, $fileTargetPath, $fileX86, $isToplevel);
842 ($isToplevel, $filePPC, $fileX86, $fileTargetPath) = @_;
844 my ($typePPC, $typeX86);
845 if (defined($filePPC) && !defined($typePPC = $filePPC->lstatType())) {
846 return complain
(1, 'makeUniversal: lstat ppc: '.$!.' for:',
849 if (defined($fileX86) && !defined($typeX86 = $fileX86->lstatType())) {
850 return complain
(1, 'makeUniversal: lstat x86: '.$!.' for:',
854 if (defined($filePPC) && defined($fileX86) && $typePPC != $typeX86) {
855 return complain
(1, 'makeUniversal: incompatible types:',
860 # $aSourceFile will contain a FileAttrCache object that will return
861 # the correct type data. It's used because it's possible for one of
862 # the two source files to be undefined (indicating a straight copy).
864 if (defined($filePPC)) {
865 $aSourceFile = $filePPC;
868 $aSourceFile = $fileX86;
871 if ($aSourceFile->lIsDir()) {
872 if ($gVerbosity >= 3 || $gDryRun) {
873 print('mkdir '.(argumentEscape
($fileTargetPath))[0]."\n");
875 if (!$gDryRun && !mkdir($fileTargetPath)) {
876 return complain
(1, 'makeUniversal: mkdir: '.$!.' for:',
882 if (!($rv = makeUniversalDirectory
($filePPC, $fileX86, $fileTargetPath))) {
883 # makeUniversalDirectory printed an error.
885 command
($gConfig{'cmd_rm'},'-rf','--',$fileTargetPath);
889 # Touch the directory when leaving it. If unify is being run on an
890 # .app bundle, the .app might show up without an icon because the
891 # system might have found the .app before it was completely built.
892 # Touching it dirties it in LaunchServices' mind.
893 if ($gVerbosity >= 3) {
894 print('touch '.(argumentEscape
($fileTargetPath))[0]."\n");
896 utime(undef, undef, $fileTargetPath);
901 elsif ($aSourceFile->lIsSymLink()) {
902 my ($linkPPC, $linkX86);
903 if (defined($filePPC) && !defined($linkPPC=readlink($filePPC->path()))) {
904 return complain
(1, 'makeUniversal: readlink ppc: '.$!.' for:',
907 if (defined($fileX86) && !defined($linkX86=readlink($fileX86->path()))) {
908 return complain
(1, 'makeUniversal: readlink x86: '.$!.' for:',
911 if (defined($filePPC) && defined($fileX86) && $linkPPC ne $linkX86) {
912 return complain
(1, 'makeUniversal: symbolic links differ:',
917 # $aLink here serves the same purpose as $aSourceFile in the enclosing
918 # block: it refers to the target of the symbolic link, whether there
919 # is one valid source or two.
921 if (defined($linkPPC)) {
928 if ($gVerbosity >= 3 || $gDryRun) {
930 join(' ',argumentEscape
($aLink, $fileTargetPath))."\n");
932 if (!$gDryRun && !symlink($aLink, $fileTargetPath)) {
933 return complain
(1, 'makeUniversal: symlink: '.$!.' for:',
940 elsif($aSourceFile->lIsRegularFile()) {
941 my ($machPPC, $machX86);
942 if (!defined($filePPC) || !defined($fileX86)) {
943 # One of the source files isn't present. The right thing to do is
944 # to just copy what does exist straight over, so skip Mach-O checks.
949 if (!defined($machPPC=$filePPC->isMachO())) {
950 return complain
(1, 'makeUniversal: isFileMachO ppc failed for:',
953 if (!defined($machX86=$fileX86->isMachO())) {
954 return complain
(1, 'makeUniversal: isFileMachO x86 failed for:',
959 if ($machPPC != $machX86) {
960 return complain
(1, 'makeUniversal: variant Mach-O attributes:',
966 # makeUniversalFile will print an error if it fails.
967 return makeUniversalFile
($filePPC, $fileX86, $fileTargetPath);
970 # Regular file. copyIfIdentical will print an error if it fails.
971 return copyIfIdentical
($filePPC, $fileX86, $fileTargetPath);
974 # Special file, don't know how to handle.
975 return complain
(1, 'makeUniversal: cannot handle special file:',
982 # Give the user a hand.
985 "usage: unify <ppc-path> <x86-path> <universal-path>\n".
986 " [--dry-run] (print what would be done)\n".
987 " [--only-one <action>] (skip, copy, fail; default=copy)\n".
988 " [--verbosity <level>] (0, 1, 2, 3; default=2)\n");
992 # readZipCRCs($zipFile)
994 # $zipFile is the pathname to a zip file whose directory will be read.
995 # A reference to a hash is returned, with the member pathnames from the
996 # zip file as keys, and reasonably unique identifiers as values. The
997 # format of the values is not specified exactly, but does include the
998 # member CRCs and sizes and differentiates between files and directories.
999 # It specifically does not distinguish between modification times. On
1000 # failure, prints a message and returns undef.
1001 sub readZipCRCs
($) {
1006 $zip = Archive
::Zip
->new();
1008 if (($ze = $zip->read($zipFile)) != AZ_OK
) {
1009 complain
(1, 'readZipCRCs: read error '.$ze.' for:',
1014 my ($member, %memberCRCs, @memberList);
1016 @memberList = $zip->members();
1018 foreach $member (@memberList) {
1019 # Take a few of the attributes that identify the file and stuff them into
1020 # the members hash. Directories will show up with size 0 and crc32 0,
1021 # so isDirectory() is used to distinguish them from empty files.
1022 $memberCRCs{$member->fileName()} = join(',', $member->isDirectory() ?
1 : 0,
1023 $member->uncompressedSize(),
1024 $member->crc32String());
1027 return {%memberCRCs};
1031 # FileAttrCache allows various attributes about a file to be cached
1032 # so that if they are needed again after first use, no system calls
1033 # will be made and the program won't need to hit the disk.
1035 package FileAttrCache
;
1037 use Fcntl
(':DEFAULT', ':mode');
1039 # FileAttrCache->new($path)
1041 # Creates a new FileAttrCache object for the file at path $path and
1042 # returns it. The cache is not primed at creation time, values are
1043 # fetched lazily as they are needed.
1045 my ($class, $path, $proto, $this);
1046 ($proto, $path) = @_;
1047 if (!($class = ref($proto))) {
1057 'magicErrMsg' => undef,
1063 bless($this, $class);
1067 # $FileAttrCache->isFat()
1069 # Returns true if the file is a fat Mach-O file, false if it's not, and
1070 # undef if an error occurs. See /usr/include/mach-o/fat.h.
1075 # magic() caches, there's no separate cache because isFat() doesn't hit
1076 # the disk other than by calling magic().
1078 if (!defined($magic = $this->magic())) {
1082 if ($magic == 0xcafebabe) {
1089 # $FileAttrCache->isMachO()
1091 # Returns true if the file is a Mach-O image (including a fat file), false
1092 # if it's not, and undef if an error occurs. See
1093 # /usr/include/mach-o/loader.h and /usr/include/mach-o/fat.h.
1098 # magic() caches, there's no separate cache because isMachO() doesn't hit
1099 # the disk other than by calling magic().
1101 if (!defined($magic = $this->magic())) {
1105 # Accept Mach-O fat files or Mach-O thin files of either endianness.
1106 if ($magic == 0xfeedface ||
1107 $magic == 0xcefaedfe ||
1108 $magic == 0xcafebabe) {
1115 # $FileAttrCache->isZip()
1117 # Returns true if the file is a zip file, false if it's not, and undef if
1118 # an error occurs. See http://www.pkware.com/business_and_developers/developer/popups/appnote.txt .
1123 # magic() caches, there's no separate cache because isFat() doesn't hit
1124 # the disk other than by calling magic().
1126 if (!defined($magic = $this->magic())) {
1130 if ($magic == 0x504b0304) {
1137 # $FileAttrCache->lIsExecutable()
1139 # Wraps $FileAttrCache->lstat(), returning true if the file is has any,
1140 # execute bit set, false if none are set, or undef if an error occurs.
1141 # On error, $! is set to lstat's errno.
1142 sub lIsExecutable
($) {
1146 if (!defined($mode = $this->lstatMode())) {
1150 return $mode & (S_IXUSR
| S_IXGRP
| S_IXOTH
);
1153 # $FileAttrCache->lIsDir()
1155 # Wraps $FileAttrCache->lstat(), returning true if the file is a directory,
1156 # false if it isn't, or undef if an error occurs. Because lstat is used,
1157 # this will return false even if the file is a symlink pointing to a
1158 # directory. On error, $! is set to lstat's errno.
1163 if (!defined($type = $this->lstatType())) {
1167 return S_ISDIR
($type);
1170 # $FileAttrCache->lIsRegularFile()
1172 # Wraps $FileAttrCache->lstat(), returning true if the file is a regular,
1173 # file, false if it isn't, or undef if an error occurs. Because lstat is
1174 # used, this will return false even if the file is a symlink pointing to a
1175 # regular file. On error, $! is set to lstat's errno.
1176 sub lIsRegularFile
($) {
1180 if (!defined($type = $this->lstatType())) {
1184 return S_ISREG
($type);
1187 # $FileAttrCache->lIsSymLink()
1189 # Wraps $FileAttrCache->lstat(), returning true if the file is a symbolic,
1190 # link, false if it isn't, or undef if an error occurs. On error, $! is
1191 # set to lstat's errno.
1196 if (!defined($type = $this->lstatType())) {
1200 return S_ISLNK
($type);
1203 # $FileAttrCache->lstat()
1205 # Wraps the lstat system call, providing a cache to speed up multiple
1206 # lstat calls for the same file. See lstat(2) and lstat in perlfunc(1).
1211 # Use the cached lstat result.
1212 if ($$this{'lstatInit'}) {
1213 if (defined($$this{'lstatErrno'})) {
1214 $! = $$this{'lstatErrno'};
1216 return @
{$$this{'lstat'}};
1218 $$this{'lstatInit'} = 1;
1220 if (!(@stat = CORE
::lstat($$this{'path'}))) {
1221 $$this{'lstatErrno'} = $!;
1224 $$this{'lstat'} = [@stat];
1228 # $FileAttrCache->lstatMode()
1230 # Wraps $FileAttrCache->lstat(), returning the mode bits from the st_mode
1231 # field, or undef if an error occurs. On error, $! is set to lstat's
1237 if (!(@stat = $this->lstat())) {
1241 return S_IMODE
($stat[2]);
1244 # $FileAttrCache->lstatType()
1246 # Wraps $FileAttrCache->lstat(), returning the type bits from the st_mode
1247 # field, or undef if an error occurs. On error, $! is set to lstat's
1253 if (!(@stat = $this->lstat())) {
1257 return S_IFMT
($stat[2]);
1260 # $FileAttrCache->magic()
1262 # Returns the "magic number" for the file by reading its first four bytes
1263 # as a big-endian unsigned 32-bit integer and returning the result. If an
1264 # error occurs, returns undef and prints diagnostic messages to stderr. If
1265 # the file is shorter than 32 bits, returns -1. A cache is provided to
1266 # speed multiple magic calls for the same file.
1271 # Use the cached magic result.
1272 if ($$this{'magicInit'}) {
1273 if (defined($$this{'magicErrno'})) {
1274 if (defined($$this{'magicErrMsg'})) {
1275 complain
(1, 'FileAttrCache::magic: '.$$this{'magicErrMsg'}.' for:',
1278 $! = $$this{'magicErrno'};
1280 return $$this{'magic'};
1283 $$this{'magicInit'} = 1;
1286 if (!sysopen($fh, $$this{'path'}, O_RDONLY
)) {
1287 $$this{'magicErrno'} = $!;
1288 $$this{'magicErrMsg'} = 'open "'.$$this{'path'}.'": '.$!;
1289 complain
(1, 'FileAttrCache::magic: '.$$this{'magicErrMsg'}.' for:',
1295 my ($bytes, $magic);
1296 if (!defined($bytes = sysread($fh, $magic, 4))) {
1297 $$this{'magicErrno'} = $!;
1298 $$this{'magicErrMsg'} = 'read "'.$$this{'path'}.'": '.$!;
1299 complain
(1, 'FileAttrCache::magic: '.$$this{'magicErrMsg'}.' for:',
1308 # The file is too short, didn't read a magic number. This isn't really
1309 # an error. Return an unlikely value.
1310 $$this{'magic'} = -1;
1314 $$this{'magic'} = unpack('N', $magic);
1315 return $$this{'magic'};
1318 # $FileAttrCache->path()
1320 # Returns the file's pathname.
1324 return $$this{'path'};
1327 # $FileAttrCache->stat()
1329 # Wraps the stat system call, providing a cache to speed up multiple
1330 # stat calls for the same file. If lstat() has already been called and
1331 # the file is not a symbolic link, the cached lstat() result will be used.
1332 # See stat(2) and lstat in perlfunc(1).
1337 # Use the cached stat result.
1338 if ($$this{'statInit'}) {
1339 if (defined($$this{'statErrno'})) {
1340 $! = $$this{'statErrno'};
1342 return @
{$$this{'stat'}};
1345 $$this{'statInit'} = 1;
1347 # If lstat has already been called, and the file isn't a symbolic link,
1348 # use the cached lstat result.
1349 if ($$this{'lstatInit'} && !$$this{'lstatErrno'} &&
1350 !S_ISLNK
(${$$this{'lstat'}}[2])) {
1351 $$this{'stat'} = $$this{'lstat'};
1352 return @
{$$this{'stat'}};
1355 if (!(@stat = CORE
::stat($$this{'path'}))) {
1356 $$this{'statErrno'} = $!;
1359 $$this{'stat'} = [@stat];
1363 # $FileAttrCache->statSize()
1365 # Wraps $FileAttrCache->stat(), returning the st_size field, or undef
1366 # undef if an error occurs. On error, $! is set to stat's errno.
1371 if (!(@stat = $this->lstat())) {