#!/usr/bin/env perl # # Parse PCI_ROM and ISA_ROM entries from source file(s) specified as # arguments and output the relevant Makefile rules to STDOUT. # # Originally based on portions of Ken Yap's genrules.pl. Completely # rewritten by Robin Smidsrød to be more maintainable. use strict; use warnings; use Getopt::Long; # Parse command-line options my @exclude_driver_classes = (); my @exclude_drivers = (); my $debug = 0; my $help = 0; GetOptions( "exclude-driver-class=s" => \@exclude_driver_classes, "exclude-driver=s" => \@exclude_drivers, "debug" => \$debug, "help" => \$help, ); # Convert exclution arrays to lookup tables my $exclude_driver_class_map = { map { $_ => 1 } @exclude_driver_classes }; my $exclude_driver_map = { map { $_ => 1 } @exclude_drivers }; # Ensure STDOUT and STDERR are synchronized if debugging if ( $debug ) { STDOUT->autoflush(1); STDERR->autoflush(1); } # Compile regular expressions here for slight performance boost my %RE = ( 'parse_driver_class' => qr{ drivers/ (\w+?) / }x, 'parse_family' => qr{^ (?:\./)? (.*) \..+? $}x, 'find_rom_line' => qr/^ \s* ( (PCI|ISA)_ROM \s* \( \s* (.*?) ) $/x, 'extract_pci_id' => qr/^ \s* 0x([0-9A-Fa-f]{4}) \s* ,? \s* (.*) $/x, 'extract_quoted_string' => qr/^ \s* \" ([^\"]*?) \" \s* ,? \s* (.*) $/x, ); # Show help if required arguments are missing or help was requested show_usage_and_exit() if $help or @ARGV < 1; # Process each source file specified process_source_file($_) for @ARGV; exit; sub show_usage_and_exit { print STDERR <<"EOM"; Syntax: $0 [] [] Options: --exclude-driver-class Exclude specified driver classes --exclude-driver Exclude specified drivers --debug Output debug information on STDERR --help This help information EOM exit 1; } # Figure out if source file is a driver and look for ROM declarations sub process_source_file { my ($source_file) = @_; return unless defined $source_file; return unless length $source_file; my $state = { 'source_file' => $source_file }; log_debug("SOURCE_FILE", $state->{source_file}); # Skip source files that aren't drivers parse_driver_class( $state ); unless ( $state->{'driver_class'} ) { log_debug("SKIP_NOT_DRIVER", $state->{source_file} ); return; } # Skip source files with driver classes that are explicitly excluded if ( $exclude_driver_class_map->{ $state->{'driver_class'} } ) { log_debug("SKIP_EXCL_CLASS", $state->{'driver_class'} ); return; } # Skip source files without driver information parse_family( $state ); parse_driver_name( $state ); unless ( $state->{'family'} and $state->{'driver_name'} ) { log_debug("SKIP_NO_DRV_INFO", $state->{source_file} ); return; } # Skip source files with drivers that are explicitly excluded if ( $exclude_driver_map->{ $state->{'driver_name'} } ) { log_debug("SKIP_EXCL_DRV", $state->{'driver_name'} ); return; } # Iterate through lines in source files looking for ROM declarations # and # output Makefile rules open( my $fh, "<", $state->{'source_file'} ) or die "Couldn't open $state->{source_file}: $!\n"; while (<$fh>) { process_rom_decl($state, $1, $2, $3) if m/$RE{find_rom_line}/; } close($fh) or die "Couldn't close $source_file: $!\n"; return 1; } # Verify that the found ROM declaration is sane and dispatch to the right # handler depending on type sub process_rom_decl { my ($state, $rom_line, $rom_type, $rom_decl) = @_; return unless defined $rom_line; return unless length $rom_line; log_debug("ROM_LINE", $rom_line); return unless defined $rom_type; return unless length $rom_type; log_debug("ROM_TYPE", $rom_type); $state->{'type'} = lc $rom_type; return process_pci_rom($state, $rom_decl) if $rom_type eq "PCI"; return process_isa_rom($state, $rom_decl) if $rom_type eq "ISA"; return; } # Extract values from PCI_ROM declaration lines and dispatch to # Makefile rule generator sub process_pci_rom { my ($state, $decl) = @_; return unless defined $decl; return unless length $decl; (my $vendor, $decl) = extract_pci_id($decl, 'PCI_VENDOR'); (my $device, $decl) = extract_pci_id($decl, 'PCI_DEVICE'); (my $image, $decl) = extract_quoted_string($decl, 'IMAGE'); (my $desc, $decl) = extract_quoted_string($decl, 'DESCRIPTION'); if ( $vendor and $device and $image and $desc ) { print_make_rules( $state, "${vendor}${device}", $desc, $vendor, $device ); print_make_rules( $state, $image, $desc, $vendor, $device, 1 ); } else { log_debug("WARNING", "Malformed PCI_ROM macro on line $. of $state->{source_file}"); } return 1; } # Extract values from ISA_ROM declaration lines and dispatch to # Makefile rule generator sub process_isa_rom { my ($state, $decl) = @_; return unless defined $decl; return unless length $decl; (my $image, $decl) = extract_quoted_string($decl, 'IMAGE'); (my $desc, $decl) = extract_quoted_string($decl, 'DESCRIPTION'); if ( $image and $desc ) { print_make_rules( $state, $image, $desc ); } else { log_debug("WARNING", "Malformed ISA_ROM macro on line $. of $state->{source_file}"); } return 1; } # Output Makefile rules for the specified ROM declarations sub print_make_rules { my ( $state, $image, $desc, $vendor, $device, $dup ) = @_; unless ( $state->{'is_header_printed'} ) { print "# NIC\t\n"; print "# NIC\tfamily\t$state->{family}\n"; print "DRIVERS_$state->{driver_class} += $state->{driver_name}\n"; print "DRIVERS += $state->{driver_name}\n"; print "\n"; $state->{'is_header_printed'} = 1; } return if $vendor and ( $vendor eq "ffff" or $device eq "ffff" ); my $ids = $vendor ? "$vendor,$device" : "-"; print "# NIC\t$image\t$ids\t$desc\n"; print "DRIVER_$image = $state->{driver_name}\n"; print "ROM_TYPE_$image = $state->{type}\n"; print "ROM_DESCRIPTION_$image = \"$desc\"\n"; print "PCI_VENDOR_$image = 0x$vendor\n" if $vendor; print "PCI_DEVICE_$image = 0x$device\n" if $device; print "ROMS += $image\n" unless $dup; print "ROMS_$state->{driver_name} += $image\n" unless $dup; print "\n"; return 1; } # Driver class is whatever comes after the "drivers" part of the filename (relative path) sub parse_driver_class { my ($state) = @_; my $filename = $state->{'source_file'}; return unless defined $filename; return unless length $filename; if ( $filename =~ m/$RE{parse_driver_class}/ ) { log_debug("DRIVER_CLASS", $1); $state->{'driver_class'} = $1; } return; } # Family name is filename (relative path) without extension sub parse_family { my ($state) = @_; my $filename = $state->{'source_file'}; return unless defined $filename; return unless length $filename; if ( $filename =~ m/$RE{parse_family}/ ) { log_debug("FAMILY", $1); $state->{'family'} = $1; } return; } # Driver name is last part of family name sub parse_driver_name { my ($state) = @_; my $family = $state->{'family'}; return unless defined $family; return unless length $family; my @parts = split "/", $family; $state->{'driver_name'} = $parts[-1]; log_debug("DRIVER", $state->{'driver_name'}); return; } # Extract a PCI vendor/device ID e.g. 0x8086, possibly followed by a comma # Should always be 4-digit lower-case hex number sub extract_pci_id { my ($str, $label) = @_; return "", $str unless defined $str; return "", $str unless length $str; if ( $str =~ m/$RE{extract_pci_id}/ ) { my $id = lc $1; log_debug($label, $id); return $id, $2; } return "", $str; } # Extract a double-quoted string, possibly followed by a comma sub extract_quoted_string { my ($str, $label) = @_; return "", $str unless defined $str; return "", $str unless length $str; if ( $str =~ m/$RE{extract_quoted_string}/ ) { log_debug($label, $1); return $1, $2; } return "", $str; } # Output debug info to STDERR (off by default) sub log_debug { my ($label, $str) = @_; return unless $debug; return unless defined $str; print STDERR "\n" if $label eq 'SOURCE_FILE'; print STDERR "="; if ( defined $label ) { my $pad_count = 16 - length $label; print STDERR $label . ":" . ( " " x $pad_count ); } print STDERR $str . "\n"; return; }