| #!/usr/bin/perl -w |
| # |
| # Program to reverse the device identifier IDs in the PCIR and PnP |
| # structures in a ROM for old non-compliant BIOSes |
| # |
| # GPL, Ken Yap 2001 |
| # |
| |
| use bytes; |
| |
| use IO::Seekable; |
| |
| sub swaplocs ($$$) |
| { |
| my ($dataref, $loc1, $loc2) = @_; |
| my ($t); |
| |
| $t = substr($$dataref, $loc1, 1); |
| substr($$dataref, $loc1, 1) = substr($$dataref, $loc2, 1); |
| substr($$dataref, $loc2, 1) = $t; |
| } |
| |
| sub printdevids ($$) |
| { |
| my ($dataref, $loc) = @_; |
| |
| return (sprintf "%02x %02x %02x", unpack('C3', substr($$dataref, $loc, 3))); |
| } |
| |
| $#ARGV >= 0 or die "Usage: $0 romimage\n"; |
| $file = $ARGV[0]; |
| open(F, "+<$file") or die "$file: $!\n"; |
| binmode(F); |
| # Handle up to 64kB ROM images |
| $len = read(F, $data, 64*1024); |
| defined($len) or die "$file: $!\n"; |
| substr($data, 0, 2) eq "\x55\xAA" or die "$file: Not a boot ROM image\n"; |
| ($pci, $pnp) = unpack('v2', substr($data, 0x18, 4)); |
| ($pci < $len and $pnp < $len) or die "$file: Not a PCI PnP ROM image\n"; |
| (substr($data, $pci, 4) eq 'PCIR' and substr($data, $pnp, 4) eq '$PnP') |
| or die "$file: No PCI and PNP structures, not a PCI PNP ROM image\n"; |
| &swaplocs(\$data, $pci+13, $pci+15); |
| &swaplocs(\$data, $pnp+18, $pnp+20); |
| seek(F, 0, SEEK_SET) or die "$file: Cannot seek to beginning\n"; |
| print F $data; |
| close(F); |
| print "PCI devids now: ", &printdevids(\$data, $pci+13), "\n"; |
| print "PnP devids now: ", &printdevids(\$data, $pnp+18), "\n"; |
| exit(0); |