#!/usr/bin/perl use FileHandle; use strict; my %depend = (); my %minver = (); my %config = (); my %intopt = (); my %hexopt = (); my %tristate = (); my %kernopts = (); my %depmods = (); my ($version, $level, $sublevel, $kernver); my $kernel = shift; my $force_kconfig = shift; my $debug=0; ########################################################### # Read a .config file (first argument) and put the resulting options in a # hash, which is returned. This subroutine should be called in a hash # context or it will not be very useful. sub process_config($) { my $filename = shift; my %conf; my $in = new FileHandle; open $in, $filename or die "File not found: $filename"; while (<$in>) { if (/^CONFIG_(\w+)\s*=\s*(\S.*?)\s*$/) { my $c = \($conf{$1} = $2); $$c =~ y/nmy/012/ if $$c =~ /^.$/; next; } if (/^# CONFIG_(\w+) is not set$/) { $conf{$1} = 0; next; } unless (/^#|$/) { print "Confused by this line in $filename:\n$_"; } } close $in; return %conf; } sub add_bool($) { my $arg=shift; exists $config{$arg} or die "Adding unknown boolean '$arg'"; print "Boolean: $arg\n" if $debug; $tristate{$arg} = "bool"; } sub add_tristate($) { my $arg=shift; exists $config{$arg} or die "Adding unknown tristate '$arg'"; print "Tristate: $arg\n" if $debug; $tristate{$arg}="tristate"; # tri default is 'm' $config{$arg} = 1 if($config{$arg}); } sub add_int($) { my $arg=shift; exists $config{$arg} or die "Adding unknown int '$arg'"; $intopt{$arg} = 0; } sub add_hex($) { my $arg=shift; exists $config{$arg} or die "Adding unknown hex '$arg'"; $hexopt{$arg} = 0; } sub set_int_value($$) { my $key = shift; my $val = shift; exists $intopt{$key} or die "Default for unknown int option '$key'"; $intopt{$key} = $val; } sub set_hex_value($$) { my $key = shift; my $val = shift; exists $hexopt{$key} or die "Default for unknown hex option '$key'"; $hexopt{$key} = "0x$val"; } sub add_config($) { my $arg = shift; if ($arg =~ /^(\w+)\s*$/) { $config{$1} = 2; } else { die "Do not understand config variable '$arg'"; } } ######################################## # Turn option off, iff it already exists sub disable_config($) { my $key = shift; $config{$key} = 0 if (exists $config{$key}); } # %{$depend{'FOO'}} lists all variables which depend on FOO. This is the # reverse of the variables that FOO depends on. my %depend = (); sub finddeps($$) { my $key = shift; my $deps = shift; $deps =~ s/^\W+//; $depend{$_}{$key} = 1 foreach(split(/\W[\Wynm]*/, $deps)); } # @{$depends{'FOO'}} is a list of dependency expressions that FOO depends # on. This is the reverse of the variables that depend on FOO. my %depends = (); sub depends($$) { my $key = shift; my $deps = shift; if(!defined $key || $key eq '') { print "Got bad key with $deps\n"; return; } finddeps($key, $deps); push @{$depends{$key}}, $deps; } sub selects($$$) { my $key = shift; my $select = shift; my $if = shift; finddeps($key, $select); if(defined $if) { finddeps($key, $if); # Transform "select X if Y" into "depends on !Y || X" $select = "!($if) || ($select)"; } push @{$depends{$key}}, $select; } # Needs: # %depend <- %depend{FOO} lists the variables that depend on FOO # %depends <- %depends{FOO} lists the dependency expressions for FOO # %config <- the value of all media variables # %kernopts <- the value of all kernel variables (including media ones) # # Checks the dependencies of all media variables, recursively. Returns # a new hash that has the new values of all media variables. sub checkdeps() { my %allconfig; @allconfig{keys %depend} = @kernopts{keys %depend}; @allconfig{keys %config} = values %config; # Set undef values to 0 map { defined $_ or $_ = 0 } values %allconfig; # First run, check deps for all the v4l-dvb variables. Following # runs, check all variables which depend on a variable that was # changed. Stop when no more variables change. for(my %changed = (), my @tocheck = keys %config; @tocheck; @tocheck = keys %changed, %changed = ()) { foreach (@tocheck) { next unless($allconfig{$_}); # skip disabled vars if (!checkvardeps($_)) { $changed{$_} = 1 foreach keys %{$depend{$_}}; } } } return map { $_ => $allconfig{$_} } keys %config; # Check the dependencies of a variable, if one fails the variable # is set to 0. Returns 0 if the variable failed, 1 if it passed. sub checkvardeps($) { my $key = shift; my $deps = $depends{$key}; foreach (@$deps) { if (!eval(toperl($_))) { $allconfig{$key} = 0; return 0; } } return 1; } # Convert a Kconfig expression to a Perl expression sub toperl($) { local $_ = shift; # Turn n,m,y into 0,1,2 s/\bn\b/0/g; s/\bm\b/1/g; s/\by\b/2/g; # Turn = into ==, but keep != as != (ie. not !==) s/(?) { # In our Kconfig files, the first non-help line after the # help text always has no indention. Technically, the # help text is ended by just by the indention decreasing, # but that is a pain to keep track of. if ($in_help && /^\S/) { $in_help = 0; } elsif ($in_help) { # Still inside help text next; } # Start of help text if (/^\s*(---)?help(---)?\s*$/) { $in_help = 1; # Insert VIDEO_KERNEL_VERSION dependency, default # n line, and help text note for disabled drivers. if ($disabled) { if(exists $tristate{$key} && !$default_seen) { print OUT "\tdefault n\n"; } print OUT "\tdepends on VIDEO_KERNEL_VERSION\n"; $_ = sprintf($disabled_msg, $minver{$key}); } next; } # No help text should get processed past this point $in_help and die "I'm very confused"; # start of a new stanza, reset if (/^\w/) { $disabled = 0; $default_seen = 0; $key = undef; } if (m|^\s*source\s+"([^"]+)"\s*$| || m|^\s*source\s+(\S+)\s*$|) { open_kconfig($dir, "$dir/$1"); $_ = ''; # don't print the source line itself next; } if (m|^\s*config (\w+)\s*$|) { $key = $1; add_config($key); if (exists $minver{$key} && cmp_ver($minver{$key}, $kernver) > 0) { $disabled = 1; disable_config($key); print "$key: Requires at least kernel $minver{$key}\n"; print OUT "# $key disabled for insufficient kernel version\n"; } else { $disabled=0; } } elsif (m|^\s*comment\s+"[^"]*"\s*$|) { $key = 'comment'; } elsif (m|^\s*menu\s+"[^"]*"\s*$|) { $key = 'menu'; } # Remaining Kconfig directives only makse sense inside Kconfig blocks next unless(defined $key); # Don't process any directives in comment blocks (or menus) next if ($key eq 'comment' || $key eq 'menu'); add_bool($key) if(/^\s*bool(ean)?\s/); add_tristate($key) if (/^\s*tristate\s/); add_int($key) if (/^\s*int\s/); add_hex($key) if (/^\s*hex\s/); depends($key, $1) if (m|^\s*depends on\s+(.+?)\s*$|); selects($key, $1, $3) if (m|^\s*select\s+(\w+)(\s+if\s+(.+?))?\s*$|); # Get default for int options if (m|^\s*default "(\d+)"| && exists $intopt{$key}) { set_int_value($key, $1); next; } # Get default for hex options if (m|^\s*default "(0x)?([[:xdigit:]]+)"| && exists $hexopt{$key}) { set_hex_value($key, $2); next; } # Override default for disabled tri/bool options # We don't care about the default for tri/bool options otherwise if (m/^\s*default (y|n|m|"yes"|"no")(\s+if .*)?\s*$/ && exists $tristate{$key} && $disabled) { $default_seen = 1; $_ = "\tdefault n\n"; next; } } continue { print OUT $_; } close $in; } sub parse_versions() { my $in = new FileHandle; my $ver; open $in, ') { if (/\[(\d+\.\d+\.\d+)\]/) { $ver = $1; } elsif (/^\s*(\w+)/) { $minver{$1} = $ver; print "minimal version for $1 is $ver\n" if $debug; } } close $in; } # Read in the .version file to get the kernel version sub get_version() { open IN, '<.version' or die 'File not found: .version'; while () { if (/KERNELRELEASE\s*[:]*[=]+\s*(\d+)\.(\d+)\.(\d+)/) { $version=$1; $level=$2; $sublevel=$3; return $kernver="$version.$level.$sublevel"; } } close IN; } # Like ver1 <=> ver2, but understands X.Y.Z version strings sub cmp_ver($$) { shift =~ /(\d+)\.(\d+)\.(\d+)/; my ($v1_ver,$v1_level,$v1_sublevel) = ($1,$2,$3); shift =~ /(\d+)\.(\d+)\.(\d+)/; my ($v2_ver,$v2_level,$v2_sublevel) = ($1,$2,$3); my $cmp = $v1_ver <=> $v2_ver; return $cmp unless($cmp == 0); $cmp = $v1_level <=> $v2_level; return $cmp unless($cmp == 0); return $v1_sublevel <=> $v2_sublevel; } # Get Kernel's config settings %kernopts = process_config("$kernel/.config"); # Modules must be on, or building out of tree drivers makes no sense if(!$kernopts{MODULES}) { print <<"EOF"; You appear to have loadable modules turned off in your kernel. You can not compile the v4l-dvb drivers, as modules, and use them with a kernel that has modules disabled. If you want to compile these drivers into your kernel, you should use 'make kernel-links' to link the source for these drivers into your kernel tree. Then configure and compile the kernel. EOF exit -1; } # Get minimum kernel version for our variables parse_versions(); # Get kernel version get_version(); print "Preparing to compile for kernel version $kernver\n"; kernelcheck(); open OUT, ">Kconfig" or die "Cannot write Kconfig file"; print OUT <<"EOF"; mainmenu "V4L/DVB menu" source Kconfig.kern config VIDEO_KERNEL_VERSION bool "Enable drivers not supported by this kernel" default n ---help--- Normally drivers that require a kernel newer $version.$level.$sublevel, the kernel you are compiling for now, will be disabled. Turning this switch on will let you enabled them, but be warned they may not work properly or even compile. They may also work fine, and the only reason they are listed as requiring a newer kernel is that no one has tested them with an older one yet. If the driver works, please post a report at V4L mailing list: video4linux-list\@redhat.com. Unless you know what you are doing, you should answer N. EOF open_kconfig('../linux', '../linux/drivers/media/Kconfig'); open_kconfig('.', './Kconfig.sound'); close OUT; # These options should default to off disable_config('DVB_AV7110_FIRMWARE'); disable_config('DVB_CINERGYT2_TUNING'); disable_config('DVB_FE_CUSTOMISE'); # Check dependencies my %newconfig = checkdeps(); # TODO: tell the user which options were disabled at this point %config = %newconfig; # Create Kconfig.kern, listing all non-media (i.e. kernel) variables # that something depended on. $depend{MODULES}{always} = 1; # Make sure MODULES will appear open OUT, '>Kconfig.kern' or die "Cannot write Kconfig.kern file: $!"; while (my ($key, $deps) = each %depend) { next if exists $config{$key}; # Skip media variables print OUT "# Needed by ", join(', ', keys %$deps), "\n"; print OUT "config $key\n\ttristate\n"; print OUT "\tdefault ", qw(n m y)[$kernopts{$key}], "\n\n"; } close OUT; # Create make dependency rules for the Kconfig file open OUT, '>.kconfig.dep' or die "Cannot write .kconfig.dep file: $!"; print OUT 'Kconfig: '; print OUT join(" \\\n\t", @kconfigfiles), "\n"; close OUT; # Produce a .config file if forced or one doesn't already exist if ($force_kconfig==1 || !-e '.config') { open OUT, '>.config' or die "Cannot write .config file: $!"; foreach (keys %tristate) { if ($config{$_}) { print OUT "CONFIG_$_=", qw(n m y)[$config{$_}], "\n"; } else { print OUT "# CONFIG_$_ is not set\n"; } } while (my ($key,$value) = each %intopt) { print OUT "CONFIG_$key=$value\n" if($config{$key}); } while (my ($key,$value) = each %hexopt) { print OUT "CONFIG_$key=$value\n" if($config{$key}); } close OUT; print "Created default (all yes) .config file\n"; } # Check for full kernel sources and print a warning sub kernelcheck() { my $fullkernel="$kernel/fs/fcntl.c"; if (! -e $fullkernel) { print <<"EOF2"; ***WARNING:*** You do not have the full kernel sources installed. This does not prevent you from building the v4l-dvb tree if you have the kernel headers, but the full kernel source may be required in order to use make menuconfig / xconfig / qconfig. If you are experiencing problems building the v4l-dvb tree, please try building against a vanilla kernel before reporting a bug. Vanilla kernels are available at http://kernel.org. On most distros, this will compile a newly downloaded kernel: cp /boot/config-`uname -r` /.config cd make all modules_install install Please see your distro's web site for instructions to build a new kernel. EOF2 } }