mirror of https://github.com/ipxe/ipxe.git
[util] Allow Option::ROM to access multiple ROM images
Signed-off-by: Michael Brown <mcb30@ipxe.org>pull/6/head
parent
cdee7866f5
commit
f2e5f8813e
|
@ -169,9 +169,11 @@ use Exporter 'import';
|
||||||
|
|
||||||
use constant ROM_SIGNATURE => 0xaa55;
|
use constant ROM_SIGNATURE => 0xaa55;
|
||||||
use constant PCI_SIGNATURE => 'PCIR';
|
use constant PCI_SIGNATURE => 'PCIR';
|
||||||
|
use constant PCI_LAST_IMAGE => 0x80;
|
||||||
use constant PNP_SIGNATURE => '$PnP';
|
use constant PNP_SIGNATURE => '$PnP';
|
||||||
|
|
||||||
our @EXPORT_OK = qw ( ROM_SIGNATURE PCI_SIGNATURE PNP_SIGNATURE );
|
our @EXPORT_OK = qw ( ROM_SIGNATURE PCI_SIGNATURE PCI_LAST_IMAGE
|
||||||
|
PNP_SIGNATURE );
|
||||||
our %EXPORT_TAGS = ( all => [ @EXPORT_OK ] );
|
our %EXPORT_TAGS = ( all => [ @EXPORT_OK ] );
|
||||||
|
|
||||||
use constant JMP_SHORT => 0xeb;
|
use constant JMP_SHORT => 0xeb;
|
||||||
|
@ -241,6 +243,53 @@ sub new {
|
||||||
|
|
||||||
=pod
|
=pod
|
||||||
|
|
||||||
|
=item C<< set ( $data ) >>
|
||||||
|
|
||||||
|
Set option ROM contents.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub set {
|
||||||
|
my $hash = shift;
|
||||||
|
my $self = tied(%$hash);
|
||||||
|
my $data = shift;
|
||||||
|
|
||||||
|
# Store data
|
||||||
|
$self->{data} = \$data;
|
||||||
|
|
||||||
|
# Split out any data belonging to the next image
|
||||||
|
delete $self->{next_image};
|
||||||
|
my $length = ( $hash->{length} * 512 );
|
||||||
|
my $pci_header = $hash->pci_header();
|
||||||
|
if ( ( $length < length $data ) &&
|
||||||
|
( defined $pci_header ) &&
|
||||||
|
( ! ( $pci_header->{last_image} & PCI_LAST_IMAGE ) ) ) {
|
||||||
|
my $remainder = substr ( $data, $length );
|
||||||
|
$data = substr ( $data, 0, $length );
|
||||||
|
$self->{next_image} = new Option::ROM;
|
||||||
|
$self->{next_image}->set ( $remainder );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
|
=item C<< get () >>
|
||||||
|
|
||||||
|
Get option ROM contents.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub get {
|
||||||
|
my $hash = shift;
|
||||||
|
my $self = tied(%$hash);
|
||||||
|
|
||||||
|
my $data = ${$self->{data}};
|
||||||
|
$data .= $self->{next_image}->get() if $self->{next_image};
|
||||||
|
return $data;
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
=item C<< load ( $filename ) >>
|
=item C<< load ( $filename ) >>
|
||||||
|
|
||||||
Load option ROM contents from the file C<$filename>.
|
Load option ROM contents from the file C<$filename>.
|
||||||
|
@ -256,8 +305,8 @@ sub load {
|
||||||
|
|
||||||
open my $fh, "<$filename"
|
open my $fh, "<$filename"
|
||||||
or croak "Cannot open $filename for reading: $!";
|
or croak "Cannot open $filename for reading: $!";
|
||||||
read $fh, my $data, ( 128 * 1024 ); # 128kB is theoretical max size
|
read $fh, my $data, -s $fh;
|
||||||
$self->{data} = \$data;
|
$hash->set ( $data );
|
||||||
close $fh;
|
close $fh;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -279,7 +328,8 @@ sub save {
|
||||||
|
|
||||||
open my $fh, ">$filename"
|
open my $fh, ">$filename"
|
||||||
or croak "Cannot open $filename for writing: $!";
|
or croak "Cannot open $filename for writing: $!";
|
||||||
print $fh ${$self->{data}};
|
my $data = $hash->get();
|
||||||
|
print $fh $data;
|
||||||
close $fh;
|
close $fh;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -339,6 +389,22 @@ sub pnp_header {
|
||||||
|
|
||||||
=pod
|
=pod
|
||||||
|
|
||||||
|
=item C<< next_image () >>
|
||||||
|
|
||||||
|
Return a C<Option::ROM> object representing the next image within the
|
||||||
|
ROM, if present.
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub next_image {
|
||||||
|
my $hash = shift;
|
||||||
|
my $self = tied(%$hash);
|
||||||
|
|
||||||
|
return $self->{next_image};
|
||||||
|
}
|
||||||
|
|
||||||
|
=pod
|
||||||
|
|
||||||
=item C<< checksum () >>
|
=item C<< checksum () >>
|
||||||
|
|
||||||
Calculate the byte checksum of the ROM.
|
Calculate the byte checksum of the ROM.
|
||||||
|
|
|
@ -27,26 +27,29 @@ my $romfile = shift || "-";
|
||||||
my $rom = new Option::ROM;
|
my $rom = new Option::ROM;
|
||||||
$rom->load ( $romfile );
|
$rom->load ( $romfile );
|
||||||
|
|
||||||
die "Not an option ROM image\n"
|
do {
|
||||||
|
|
||||||
|
die "Not an option ROM image\n"
|
||||||
unless $rom->{signature} == ROM_SIGNATURE;
|
unless $rom->{signature} == ROM_SIGNATURE;
|
||||||
|
|
||||||
my $romlength = ( $rom->{length} * 512 );
|
my $romlength = ( $rom->{length} * 512 );
|
||||||
my $filelength = $rom->length;
|
my $filelength = $rom->length;
|
||||||
die "ROM image truncated (is $filelength, should be $romlength)\n"
|
die "ROM image truncated (is $filelength, should be $romlength)\n"
|
||||||
if $filelength < $romlength;
|
if $filelength < $romlength;
|
||||||
|
|
||||||
printf "ROM header:\n\n";
|
printf "ROM header:\n\n";
|
||||||
printf " %-16s 0x%02x (%d)\n", "Length:", $rom->{length}, ( $rom->{length} * 512 );
|
printf " %-16s 0x%02x (%d)\n", "Length:",
|
||||||
printf " %-16s 0x%02x (%s0x%02x)\n", "Checksum:", $rom->{checksum},
|
$rom->{length}, ( $rom->{length} * 512 );
|
||||||
|
printf " %-16s 0x%02x (%s0x%02x)\n", "Checksum:", $rom->{checksum},
|
||||||
( ( $rom->checksum == 0 ) ? "" : "INCORRECT: " ), $rom->checksum;
|
( ( $rom->checksum == 0 ) ? "" : "INCORRECT: " ), $rom->checksum;
|
||||||
printf " %-16s 0x%04x\n", "Init:", $rom->{init};
|
printf " %-16s 0x%04x\n", "Init:", $rom->{init};
|
||||||
printf " %-16s 0x%04x\n", "UNDI header:", $rom->{undi_header};
|
printf " %-16s 0x%04x\n", "UNDI header:", $rom->{undi_header};
|
||||||
printf " %-16s 0x%04x\n", "PCI header:", $rom->{pci_header};
|
printf " %-16s 0x%04x\n", "PCI header:", $rom->{pci_header};
|
||||||
printf " %-16s 0x%04x\n", "PnP header:", $rom->{pnp_header};
|
printf " %-16s 0x%04x\n", "PnP header:", $rom->{pnp_header};
|
||||||
printf "\n";
|
printf "\n";
|
||||||
|
|
||||||
my $pci = $rom->pci_header();
|
my $pci = $rom->pci_header();
|
||||||
if ( $pci ) {
|
if ( $pci ) {
|
||||||
printf "PCI header:\n\n";
|
printf "PCI header:\n\n";
|
||||||
printf " %-16s %s\n", "Signature:", $pci->{signature};
|
printf " %-16s %s\n", "Signature:", $pci->{signature};
|
||||||
printf " %-16s 0x%04x\n", "Vendor ID:", $pci->{vendor_id};
|
printf " %-16s 0x%04x\n", "Vendor ID:", $pci->{vendor_id};
|
||||||
|
@ -57,15 +60,16 @@ if ( $pci ) {
|
||||||
$pci->{image_length}, ( $pci->{image_length} * 512 );
|
$pci->{image_length}, ( $pci->{image_length} * 512 );
|
||||||
printf " %-16s 0x%04x (%d)\n", "Runtime length:",
|
printf " %-16s 0x%04x (%d)\n", "Runtime length:",
|
||||||
$pci->{runtime_length}, ( $pci->{runtime_length} * 512 );
|
$pci->{runtime_length}, ( $pci->{runtime_length} * 512 );
|
||||||
|
printf " %-16s 0x%02x\n", "Code type:", $pci->{code_type};
|
||||||
if ( exists $pci->{conf_header} ) {
|
if ( exists $pci->{conf_header} ) {
|
||||||
printf " %-16s 0x%04x\n", "Config header:", $pci->{conf_header};
|
printf " %-16s 0x%04x\n", "Config header:", $pci->{conf_header};
|
||||||
printf " %-16s 0x%04x\n", "CLP entry:", $pci->{clp_entry};
|
printf " %-16s 0x%04x\n", "CLP entry:", $pci->{clp_entry};
|
||||||
}
|
}
|
||||||
printf "\n";
|
printf "\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
my $pnp = $rom->pnp_header();
|
my $pnp = $rom->pnp_header();
|
||||||
if ( $pnp ) {
|
if ( $pnp ) {
|
||||||
printf "PnP header:\n\n";
|
printf "PnP header:\n\n";
|
||||||
printf " %-16s %s\n", "Signature:", $pnp->{signature};
|
printf " %-16s %s\n", "Signature:", $pnp->{signature};
|
||||||
printf " %-16s 0x%02x (%s0x%02x)\n", "Checksum:", $pnp->{checksum},
|
printf " %-16s 0x%02x (%s0x%02x)\n", "Checksum:", $pnp->{checksum},
|
||||||
|
@ -78,4 +82,6 @@ if ( $pnp ) {
|
||||||
printf " %-16s 0x%04x\n", "BDV:", $pnp->{bdv};
|
printf " %-16s 0x%04x\n", "BDV:", $pnp->{bdv};
|
||||||
printf " %-16s 0x%04x\n", "BEV:", $pnp->{bev};
|
printf " %-16s 0x%04x\n", "BEV:", $pnp->{bev};
|
||||||
printf "\n";
|
printf "\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
} while ( $rom = $rom->next_image );
|
||||||
|
|
|
@ -28,7 +28,11 @@ my @romfiles = @ARGV;
|
||||||
foreach my $romfile ( @romfiles ) {
|
foreach my $romfile ( @romfiles ) {
|
||||||
my $rom = new Option::ROM;
|
my $rom = new Option::ROM;
|
||||||
$rom->load ( $romfile );
|
$rom->load ( $romfile );
|
||||||
$rom->pnp_header->fix_checksum() if $rom->pnp_header;
|
my $image = $rom;
|
||||||
$rom->fix_checksum();
|
while ( $image ) {
|
||||||
|
$image->pnp_header->fix_checksum() if $image->pnp_header;
|
||||||
|
$image->fix_checksum();
|
||||||
|
$image = $image->next_image();
|
||||||
|
}
|
||||||
$rom->save ( $romfile );
|
$rom->save ( $romfile );
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue