2013-08-20 23:04:15 -07:00
|
|
|
|
#!/usr/bin/perl -w
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
=head1 NAME
|
|
|
|
|
|
|
|
|
|
debinhex.pl - use Convert::BinHex to decode BinHex files
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
=head1 USAGE
|
|
|
|
|
|
|
|
|
|
Usage:
|
|
|
|
|
|
|
|
|
|
debinhex.pl [options] file ... file
|
|
|
|
|
|
|
|
|
|
Where the options are:
|
|
|
|
|
|
|
|
|
|
-o dir Output in given directory (default outputs in file's directory)
|
|
|
|
|
-v Verbose output (normally just one line per file is shown)
|
|
|
|
|
|
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
|
|
|
|
|
|
Each file is expected to be a BinHex file. By default, the output file is
|
|
|
|
|
given the name that the BinHex file dictates, regardless of the name of
|
|
|
|
|
the BinHex file.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
=head1 WARNINGS
|
|
|
|
|
|
|
|
|
|
Largely untested.
|
|
|
|
|
|
|
|
|
|
|
2013-08-20 23:56:25 -07:00
|
|
|
|
=head1 AUTHORS
|
2013-08-20 23:04:15 -07:00
|
|
|
|
|
|
|
|
|
Paul J. Schinder (NASA/GSFC) mostly, though Eryq can't seem to keep
|
|
|
|
|
his grubby paws off anything...
|
|
|
|
|
|
2013-08-20 23:56:25 -07:00
|
|
|
|
S<EFBFBD>ren M. Andersen (somian), made it actually work under Perl 5.8.7 on MSWin32.
|
2013-08-20 23:04:15 -07:00
|
|
|
|
|
|
|
|
|
=cut
|
|
|
|
|
|
2013-08-23 07:53:14 -07:00
|
|
|
|
# VERSION
|
|
|
|
|
|
2013-08-20 23:56:25 -07:00
|
|
|
|
my $The_OS;
|
|
|
|
|
BEGIN { $The_OS = $^O ? $^O : q// }
|
|
|
|
|
eval { require Mac::Files } if ($The_OS eq "MacOS");
|
2013-08-20 23:04:15 -07:00
|
|
|
|
|
|
|
|
|
use Getopt::Std;
|
|
|
|
|
use Convert::BinHex;
|
|
|
|
|
use POSIX;
|
|
|
|
|
use Fcntl;
|
|
|
|
|
use File::Basename;
|
|
|
|
|
use Carp;
|
|
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
|
use vars qw(
|
|
|
|
|
$opt_o
|
|
|
|
|
$opt_v
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
my $DEBUG = 0;
|
|
|
|
|
|
|
|
|
|
#------------------------------------------------------------
|
|
|
|
|
# main
|
|
|
|
|
#------------------------------------------------------------
|
|
|
|
|
sub main {
|
|
|
|
|
|
|
|
|
|
# What usage?
|
|
|
|
|
@ARGV or usage();
|
|
|
|
|
getopts('o:v');
|
|
|
|
|
$DEBUG = $opt_v;
|
|
|
|
|
|
|
|
|
|
# Process files:
|
|
|
|
|
my $file;
|
|
|
|
|
foreach $file (@ARGV) {
|
|
|
|
|
debinhex($file);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
exit(&main ? 0 : -1);
|
|
|
|
|
|
|
|
|
|
#------------------------------------------------------------
|
|
|
|
|
# usage
|
|
|
|
|
#------------------------------------------------------------
|
|
|
|
|
# Get usage from me.
|
|
|
|
|
|
|
|
|
|
sub usage {
|
|
|
|
|
my $msg = shift || '';
|
|
|
|
|
my $usage = '';
|
|
|
|
|
if (open(USAGE, "<$0")) {
|
2013-08-20 23:56:25 -07:00
|
|
|
|
while (defined($_ = <USAGE>) and !/^=head1 USAGE/) {};
|
|
|
|
|
while (defined($_ = <USAGE>) and !/^=head1/) {$usage .= $_};
|
2013-08-20 23:04:15 -07:00
|
|
|
|
close USAGE;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
$usage = "Usage unavailable; please see the script itself.";
|
|
|
|
|
}
|
|
|
|
|
print STDERR "\n$msg$usage";
|
|
|
|
|
exit -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#------------------------------------------------------------
|
|
|
|
|
# debinhex FILE
|
|
|
|
|
#------------------------------------------------------------
|
|
|
|
|
# Decode the given FILE.
|
|
|
|
|
#
|
|
|
|
|
sub debinhex {
|
|
|
|
|
my $inpath = shift || croak("No filename given $!");
|
|
|
|
|
local *BHEX;
|
|
|
|
|
my ($data, $testlength, $length, $fd);
|
|
|
|
|
|
|
|
|
|
print "DeBinHexing: $inpath\n";
|
|
|
|
|
|
|
|
|
|
# Open BinHex file:
|
|
|
|
|
open(BHEX,"<$inpath") || croak("Unable to open $inpath: $!");
|
2013-08-20 23:56:25 -07:00
|
|
|
|
binmode BHEX;
|
2013-08-20 23:04:15 -07:00
|
|
|
|
|
|
|
|
|
# Create converter interface on stream:
|
|
|
|
|
my $hqx = Convert::BinHex->open(FH => \*BHEX);
|
|
|
|
|
|
|
|
|
|
# Read header, and output as string if debugging:
|
|
|
|
|
$hqx->read_header;
|
|
|
|
|
print $hqx->header_as_string if $DEBUG;
|
|
|
|
|
|
|
|
|
|
# Get output directory/filename:
|
|
|
|
|
my ($inname, $indir) = fileparse($inpath);
|
|
|
|
|
my $outname = $hqx->filename || 'NONAME';
|
|
|
|
|
my $outdir = $opt_o || $indir;
|
|
|
|
|
my $outpath = "$outdir/$outname"; $outpath =~ s{/+}{/}g;
|
|
|
|
|
|
|
|
|
|
# Create Mac file:
|
2013-08-20 23:56:25 -07:00
|
|
|
|
if ($The_OS eq "MacOS") {
|
2013-08-20 23:04:15 -07:00
|
|
|
|
Mac::Files::FSpCreate($outpath, $hqx->creator, $hqx->type)
|
|
|
|
|
or croak("Unable to create Mac file $outpath");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# Get lengths of forks:
|
|
|
|
|
my $dlength = $hqx->data_length;
|
|
|
|
|
my $rlength = $hqx->resource_length;
|
|
|
|
|
|
|
|
|
|
# Write data fork:
|
|
|
|
|
print "Writing: $outpath\n";
|
2013-08-20 23:56:25 -07:00
|
|
|
|
$fd = POSIX::open($outpath, (&POSIX::O_WRONLY | &POSIX::O_CREAT | &Fcntl::O_BINARY), 0755);
|
2013-08-20 23:04:15 -07:00
|
|
|
|
$testlength = 0;
|
|
|
|
|
while (defined($data = $hqx->read_data)) {
|
|
|
|
|
$length = length($data);
|
|
|
|
|
POSIX::write($fd, $data, $length)
|
|
|
|
|
or croak("couldn't write $length bytes: $!");
|
|
|
|
|
$testlength += $length;
|
|
|
|
|
}
|
|
|
|
|
POSIX::close($fd) or croak "Unable to close $outpath";
|
|
|
|
|
croak("Data fork length mismatch: ".
|
|
|
|
|
"expected $dlength, wrote $testlength")
|
|
|
|
|
if $dlength != $testlength;
|
|
|
|
|
|
|
|
|
|
# Write resource fork?
|
|
|
|
|
if ($rlength) {
|
|
|
|
|
|
|
|
|
|
# Determine how to open fork file appropriately:
|
|
|
|
|
my ($rpath, $rflags);
|
2013-08-20 23:56:25 -07:00
|
|
|
|
if ($The_OS eq "MacOS") {
|
2013-08-20 23:04:15 -07:00
|
|
|
|
$rpath = $outpath;
|
|
|
|
|
$rflags = (&POSIX::O_WRONLY | &POSIX::O_CREAT | &Fcntl::O_RSRC);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
$rpath = "$outpath.rsrc";
|
2013-08-20 23:56:25 -07:00
|
|
|
|
$rflags = (&POSIX::O_WRONLY | &POSIX::O_CREAT | &Fcntl::O_BINARY);
|
2013-08-20 23:04:15 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# Write resource fork...
|
|
|
|
|
$fd = POSIX::open($rpath, $rflags, 0755);
|
|
|
|
|
$testlength = 0;
|
|
|
|
|
while (defined($data = $hqx->read_resource)) {
|
|
|
|
|
$length = length($data);
|
|
|
|
|
POSIX::write($fd,$data,$length)
|
|
|
|
|
or croak "Couldn't write $length bytes: $!";
|
|
|
|
|
$testlength += $length;
|
|
|
|
|
}
|
|
|
|
|
POSIX::close($fd) or croak "Unable to close $rpath";
|
|
|
|
|
croak("Resource fork length mismatch: ".
|
|
|
|
|
"expected $rlength, wrote $testlength")
|
|
|
|
|
if $testlength != $rlength;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# Set Mac attributes:
|
2013-08-20 23:56:25 -07:00
|
|
|
|
if ($The_OS eq "MacOS") {
|
2013-08-20 23:04:15 -07:00
|
|
|
|
my $has = Mac::Files::FSpGetCatInfo($outpath);
|
|
|
|
|
my $finfo = $has->{ioFlFndrInfo};
|
|
|
|
|
$finfo->{fdFlags} = $hqx->flags & 0xfeff; #turn off inited bit
|
|
|
|
|
$finfo->{fdType} = $hqx->type || "????";
|
|
|
|
|
$finfo->{fdCreator} = $hqx->creator || "????";
|
|
|
|
|
|
|
|
|
|
# Turn on the bundle bit if it's an application:
|
|
|
|
|
### $finfo->{fdFlags} |= 0x2000 if $finfo->{fdType} eq "APPL";
|
|
|
|
|
|
|
|
|
|
if ($DEBUG) {
|
|
|
|
|
printf("%x\n",$finfo->{fdFlags});
|
|
|
|
|
printf("%s\n",$finfo->{fdType});
|
|
|
|
|
printf("%s\n",$finfo->{fdCreator});
|
|
|
|
|
}
|
|
|
|
|
$has->{ioFlFndrInfo} = $finfo;
|
|
|
|
|
Mac::Files::FSpSetCatInfo($outpath, $has)
|
|
|
|
|
or croak "Unable to set catalog info $^E";
|
|
|
|
|
if ($DEBUG) {
|
|
|
|
|
$has = Mac::Files::FSpGetCatInfo ($outpath);
|
|
|
|
|
printf("%x\n",$has->{ioFlFndrInfo}->{fdFlags});
|
|
|
|
|
printf("%s\n",$has->{ioFlFndrInfo}->{fdType});
|
|
|
|
|
printf("%s\n",$has->{ioFlFndrInfo}->{fdCreator});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#------------------------------------------------------------
|
2013-08-20 23:56:25 -07:00
|
|
|
|
__END__
|
|
|
|
|
# Last modified: 16 Feb 2006 at 05:16 PM EST
|