| #!/usr/bin/perl -w |
| # |
| # Copyright (C) 2008 Michael Brown <mbrown@fensystems.co.uk>. |
| # |
| # This program is free software; you can redistribute it and/or |
| # modify it under the terms of the GNU General Public License as |
| # published by the Free Software Foundation; either version 2 of the |
| # License, or any later version. |
| # |
| # This program is distributed in the hope that it will be useful, but |
| # WITHOUT ANY WARRANTY; without even the implied warranty of |
| # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| # General Public License for more details. |
| # |
| # You should have received a copy of the GNU General Public License |
| # along with this program; if not, write to the Free Software |
| # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA |
| # 02110-1301, USA. |
| |
| use strict; |
| use warnings; |
| |
| use FindBin; |
| use lib "$FindBin::Bin"; |
| use Option::ROM qw ( :all ); |
| |
| sub merge_entry_points { |
| my $baserom_entry = \shift; |
| my $rom_entry = \shift; |
| my $offset = shift; |
| |
| if ( $$rom_entry ) { |
| my $old_entry = $$baserom_entry; |
| $$baserom_entry = ( $offset + $$rom_entry ); |
| $$rom_entry = $old_entry; |
| } |
| } |
| |
| my @romfiles = @ARGV; |
| my @roms = map { my $rom = new Option::ROM; $rom->load($_); $rom } @romfiles; |
| |
| my $baserom = shift @roms; |
| my $offset = $baserom->length; |
| |
| foreach my $rom ( @roms ) { |
| |
| # Merge initialisation entry point |
| merge_entry_points ( $baserom->{init}, $rom->{init}, $offset ); |
| |
| # Merge BOFM header |
| merge_entry_points ( $baserom->{bofm_header}, $rom->{bofm_header}, $offset ); |
| |
| # Update PCI header, if present in both |
| my $baserom_pci = $baserom->pci_header; |
| my $rom_pci = $rom->pci_header; |
| if ( $baserom_pci && $rom_pci ) { |
| |
| # Update PCI lengths |
| $baserom_pci->{image_length} += $rom_pci->{image_length}; |
| if ( exists $baserom_pci->{runtime_length} ) { |
| if ( exists $rom_pci->{runtime_length} ) { |
| $baserom_pci->{runtime_length} += $rom_pci->{runtime_length}; |
| } else { |
| $baserom_pci->{runtime_length} += $rom_pci->{image_length}; |
| } |
| } |
| |
| # Merge CLP entry point |
| if ( exists ( $baserom_pci->{clp_entry} ) && |
| exists ( $rom_pci->{clp_entry} ) ) { |
| merge_entry_points ( $baserom_pci->{clp_entry}, $rom_pci->{clp_entry}, |
| $offset ); |
| } |
| } |
| |
| # Update PnP header, if present in both |
| my $baserom_pnp = $baserom->pnp_header; |
| my $rom_pnp = $rom->pnp_header; |
| if ( $baserom_pnp && $rom_pnp ) { |
| merge_entry_points ( $baserom_pnp->{bcv}, $rom_pnp->{bcv}, $offset ); |
| merge_entry_points ( $baserom_pnp->{bdv}, $rom_pnp->{bdv}, $offset ); |
| merge_entry_points ( $baserom_pnp->{bev}, $rom_pnp->{bev}, $offset ); |
| } |
| |
| # Update iPXE header, if present |
| my $baserom_ipxe = $baserom->ipxe_header; |
| my $rom_ipxe = $rom->ipxe_header; |
| if ( $baserom_ipxe ) { |
| |
| # Update shrunk length |
| $baserom_ipxe->{shrunk_length} = ( $baserom->{length} + |
| ( $rom_ipxe ? |
| $rom_ipxe->{shrunk_length} : |
| $rom->{length} ) ); |
| |
| # Fix checksum |
| $baserom_ipxe->fix_checksum(); |
| } |
| |
| # Update base length |
| $baserom->{length} += $rom->{length}; |
| |
| # Fix checksum for this ROM segment |
| $rom->fix_checksum(); |
| |
| # Add this ROM to base ROM |
| my $data = substr ( $baserom->get(), 0, $baserom->length() ); |
| $data .= $rom->get(); |
| $data .= $baserom->next_image()->get() if $baserom->next_image(); |
| $baserom->set ( $data ); |
| |
| $offset += $rom->length; |
| } |
| |
| $baserom->pnp_header->fix_checksum() if $baserom->pnp_header; |
| $baserom->fix_checksum(); |
| $baserom->save ( "-" ); |