2016-04-22 06:50:48 +00:00
|
|
|
#!/usr/bin/env perl
|
|
|
|
use warnings;
|
|
|
|
use strict;
|
|
|
|
#
|
|
|
|
# Apply a diff of two hexdumps as a binary patch
|
2016-04-23 04:02:28 +00:00
|
|
|
#
|
|
|
|
# Copyright (C) 2016 Hamish Coleman
|
2016-04-22 06:50:48 +00:00
|
|
|
|
|
|
|
use IO::File;
|
|
|
|
|
|
|
|
use Data::Dumper;
|
|
|
|
$Data::Dumper::Indent = 1;
|
|
|
|
$Data::Dumper::Sortkeys = 1;
|
|
|
|
$Data::Dumper::Quotekeys = 0;
|
|
|
|
|
|
|
|
sub usage() {
|
|
|
|
print("This utility will take a diff of two hexdumps and can apply\n");
|
|
|
|
print("the binary changes to the original binary file, using the\n");
|
|
|
|
print("context lines to provide data to match and confirm we are\n");
|
|
|
|
print("patching the right file\n");
|
|
|
|
print("\n");
|
|
|
|
print("Usage:\n");
|
2017-06-20 07:58:55 +00:00
|
|
|
print(" hexpatch.pl [--rm_on_fail] binaryfile patchfile [patchfile...]\n");
|
2016-04-22 06:50:48 +00:00
|
|
|
print("\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
sub parse_hexline {
|
|
|
|
my $line = shift;
|
|
|
|
|
|
|
|
my $addr;
|
|
|
|
my $binary;
|
|
|
|
|
|
|
|
my @fields = split(/\s+/,$line);
|
|
|
|
|
|
|
|
if ($fields[0] !~ m/^([0-9a-fA-F])+$/) {
|
|
|
|
return undef;
|
|
|
|
}
|
|
|
|
$addr = hex($fields[0]);
|
|
|
|
shift @fields;
|
|
|
|
|
|
|
|
while (defined($fields[0]) && $fields[0] =~ m/^([0-9a-fA-F])+$/) {
|
|
|
|
$binary .= chr(hex($fields[0]));
|
|
|
|
shift @fields;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ($addr,$binary);
|
|
|
|
}
|
|
|
|
|
|
|
|
sub read_patchfile {
|
|
|
|
my $filename = shift;
|
|
|
|
|
|
|
|
my $fh = IO::File->new($filename, O_RDONLY);
|
|
|
|
if (!defined($fh)) {
|
|
|
|
return undef;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $db = {};
|
|
|
|
|
|
|
|
my $seen_at_symbols = 0;
|
|
|
|
|
|
|
|
while(<$fh>) {
|
|
|
|
# anything before we see a starting at symbol line can be ignored
|
|
|
|
if (!$seen_at_symbols) {
|
2016-04-22 07:23:34 +00:00
|
|
|
if (m/^\@\@ (.*) \@\@$/) {
|
2016-04-22 06:50:48 +00:00
|
|
|
$seen_at_symbols = 1;
|
2016-04-22 07:23:34 +00:00
|
|
|
$db->{name} = $1;
|
2016-04-22 06:50:48 +00:00
|
|
|
}
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $addr;
|
|
|
|
my $binary;
|
|
|
|
my $type;
|
|
|
|
if (m/^([-+ ])(.*)/) {
|
|
|
|
# a context, oldfile or newfile line
|
|
|
|
$type = $1;
|
|
|
|
($addr, $binary) = parse_hexline($2);
|
|
|
|
} elsif (m/^\@\@ .* \@\@$/) {
|
|
|
|
# hunk separator - we can ignore it since we use the hex addr
|
|
|
|
next;
|
|
|
|
} else {
|
|
|
|
warn("Invalid prefix in patchfile");
|
|
|
|
# TODO - linenumber
|
|
|
|
return(undef);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!defined($addr)) {
|
|
|
|
warn("Invalid data in hexline");
|
|
|
|
# TODO - linenumber
|
|
|
|
return(undef);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!defined($binary)) {
|
|
|
|
# this line was empty of data
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($type eq ' ') {
|
|
|
|
$type = 'context';
|
|
|
|
} elsif ($type eq '-') {
|
|
|
|
$type = 'old';
|
|
|
|
} elsif ($type eq '+') {
|
|
|
|
$type = 'new';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (defined($db->{addr}{$addr}{$type})) {
|
|
|
|
warn("Duplicate address in hexdata");
|
|
|
|
# TODO - linenumber
|
|
|
|
return(undef);
|
|
|
|
}
|
|
|
|
|
|
|
|
$db->{addr}{$addr}{$type} = $binary;
|
|
|
|
}
|
|
|
|
return $db;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub verify_context {
|
|
|
|
my $db = shift;
|
|
|
|
my $fh = shift;
|
|
|
|
|
|
|
|
for my $addr (sort keys(%{$db->{addr}})) {
|
|
|
|
my $entry = $db->{addr}{$addr};
|
|
|
|
my $expected;
|
|
|
|
if ($entry->{context}) {
|
|
|
|
$expected = $entry->{context};
|
|
|
|
} elsif ($entry->{old}) {
|
|
|
|
$expected = $entry->{old};
|
|
|
|
} elsif ($entry->{new}) {
|
|
|
|
warn("Address $addr has new data but no old data\n");
|
|
|
|
return undef;
|
|
|
|
} else {
|
|
|
|
# No data of any kind here, just skip it
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $found;
|
|
|
|
$fh->seek($addr,SEEK_SET);
|
|
|
|
$fh->read($found,length($expected));
|
|
|
|
|
|
|
|
if ($found ne $expected) {
|
|
|
|
warn("Address $addr mismatched data\n");
|
|
|
|
return undef;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub apply_patch {
|
|
|
|
my $db = shift;
|
|
|
|
my $fh = shift;
|
|
|
|
|
|
|
|
for my $addr (sort keys(%{$db->{addr}})) {
|
|
|
|
my $entry = $db->{addr}{$addr};
|
|
|
|
|
|
|
|
my $newdata;
|
|
|
|
if ($entry->{new}) {
|
|
|
|
$newdata = $entry->{new};
|
|
|
|
} else {
|
|
|
|
# No data to write at this address, just skip it
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
|
|
|
|
$fh->seek($addr,SEEK_SET);
|
|
|
|
$fh->print($newdata);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-20 07:58:55 +00:00
|
|
|
sub rm_on_fail {
|
|
|
|
my ($bool, $filename) = @_;
|
|
|
|
|
|
|
|
if ($bool) {
|
|
|
|
unlink($filename);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-22 06:50:48 +00:00
|
|
|
sub main() {
|
2017-06-20 07:58:55 +00:00
|
|
|
my $rm_on_fail;
|
|
|
|
if (defined($ARGV[0]) && $ARGV[0] eq "--rm_on_fail") {
|
|
|
|
$rm_on_fail=1;
|
|
|
|
shift @ARGV;
|
|
|
|
}
|
|
|
|
|
2016-04-22 06:50:48 +00:00
|
|
|
my $binaryfile = shift @ARGV;
|
2016-04-22 07:23:34 +00:00
|
|
|
if (!defined($binaryfile) or !defined($ARGV[0])) {
|
2016-04-22 06:50:48 +00:00
|
|
|
usage();
|
|
|
|
}
|
|
|
|
|
|
|
|
my $fh = IO::File->new($binaryfile, O_RDWR);
|
|
|
|
if (!defined($fh)) {
|
|
|
|
warn("Could not open binaryfile\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2016-04-22 07:23:34 +00:00
|
|
|
print("Attempting to patch $binaryfile\n");
|
2016-04-22 06:50:48 +00:00
|
|
|
|
2016-04-22 07:23:34 +00:00
|
|
|
while ($ARGV[0]) {
|
|
|
|
my $patchfile = shift @ARGV;
|
|
|
|
|
2016-04-23 01:50:52 +00:00
|
|
|
if (!-e $patchfile) {
|
|
|
|
warn("Patchfile $patchfile not present, skipping\n");
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
|
2016-04-22 07:23:34 +00:00
|
|
|
my $db = read_patchfile($patchfile);
|
|
|
|
|
|
|
|
if (!defined($db)) {
|
|
|
|
warn("Cannot read $patchfile\n");
|
2017-06-20 07:58:55 +00:00
|
|
|
rm_on_fail($rm_on_fail,$binaryfile);
|
2016-04-22 07:23:34 +00:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!verify_context($db,$fh)) {
|
|
|
|
warn("The binaryfile does not match the context bytes from $patchfile\n");
|
2017-06-20 07:58:55 +00:00
|
|
|
rm_on_fail($rm_on_fail,$binaryfile);
|
2016-04-22 07:23:34 +00:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
print("Applying ",$patchfile," ",$db->{name},"\n");
|
|
|
|
apply_patch($db,$fh);
|
|
|
|
}
|
2016-04-22 06:50:48 +00:00
|
|
|
}
|
|
|
|
main();
|
|
|
|
|