diff options
Diffstat (limited to 'gnulib/check-module')
m--------- | gnulib | 0 | ||||
-rwxr-xr-x | gnulib/check-module | 299 |
2 files changed, 299 insertions, 0 deletions
diff --git a/gnulib b/gnulib deleted file mode 160000 -Subproject 443bc5ffcf7429e557f4a371b0661abe98ddbc1 diff --git a/gnulib/check-module b/gnulib/check-module new file mode 100755 index 0000000..a302f1b --- /dev/null +++ b/gnulib/check-module @@ -0,0 +1,299 @@ +#!/usr/bin/perl -w +# Check a gnulib module. + +# Copyright (C) 2005-2007, 2009-2011 Free Software Foundation, Inc. + +# This file 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 3 of the License, or +# (at your option) 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, see <http://www.gnu.org/licenses/>. + + +# Read a module description file and derive the set of files +# included directly by any .c or .h file listed in the `Files:' section. +# Take the union of all such sets for any dependent modules. +# Then, compare that set with the set derived from the names +# listed in the various Files: sections. + +# This script makes no attempt to diagnose invalid or empty +# module-description files. + +# Written by Jim Meyering + +# FIXME: +# for each .m4 file listed in the Files: section(s) +# parse it for AC_LIBSOURCES directives, and accumulate the set +# of files `required' via all AC_LIBSOURCES. +# If this set is not empty, ensure that it contains +# the same (.c and .h only?) files as are listed in the Files: sections. + +use strict; +use Getopt::Long; +use File::Basename; +#use Coda; + +my $COPYRIGHT_NOTICE = "Copyright (C) 2006 Free Software Foundation, Inc.\n". +"This is free software. You may redistribute copies of it under the terms of\n". +"the GNU General Public License <http://www.gnu.org/licenses/gpl.html>.\n". +"There is NO WARRANTY, to the extent permitted by law.\n"; + +(my $VERSION = '$Revision: 1.8 $ ') =~ tr/[0-9].//cd; +(my $ME = $0) =~ s|.*/||; + +use constant ST_INIT => 1; +use constant ST_FILES => 2; +use constant ST_DEPENDENTS => 3; + +# Parse a module file (returning list of Files: names and +# list of dependent-modules. +# my ($file, $dep) = parse_module_file $module_file; +sub parse_module_file ($) +{ + my ($module_file) = @_; + + open FH, '<', $module_file + or die "$ME: can't open `$module_file' for reading: $!\n"; + + my %file_set; + my %dep_set; + + my $state = ST_INIT; + while (defined (my $line = <FH>)) + { + if ($state eq ST_INIT) + { + if ($line =~ /^Files:$/) + { + $state = ST_FILES; + } + elsif ($line =~ /^Depends-on:$/) + { + $state = ST_DEPENDENTS; + } + } + else + { + chomp $line; + $line =~ s/^\s+//; + $line =~ s/\s+$//; + if ( ! $line) + { + $state = ST_INIT; + next; + } + + if ($state eq ST_FILES) + { + $file_set{$line} = 1; + } + elsif ($state eq ST_DEPENDENTS) + { + $dep_set{$line} = 1; + (my $base = $module_file) =~ s,.*/,,; + $line eq $base + and die "$ME: module $module_file depends on itself\n"; + } + } + } + close FH; + + # my @t = sort keys %file_set; + # print "files: @t\n"; + # my @u = sort keys %dep_set; + # print "dependents: @u\n"; + + return (\%file_set, \%dep_set); +} + +# Extract the set of files required for this module, including +# those required via dependent modules. + +# Files: +# lib/stat.c +# m4/stat.m4 +# lib/foo.h +# +# Depends-on: +# some-other-module + +sub usage ($) +{ + my ($exit_code) = @_; + my $STREAM = ($exit_code == 0 ? *STDOUT : *STDERR); + if ($exit_code != 0) + { + print $STREAM "Try `$ME --help' for more information.\n"; + } + else + { + print $STREAM <<EOF; +Usage: $ME [OPTIONS] FILE... + +Read a module description file and derive the set of files +included directly by any .c or .h file listed in the `Files:' section. +Take the union of all such sets for any dependent modules. +Then, compare that set with the set derived from the names +listed in the various Files: sections. + +OPTIONS: + + --help display this help and exit + --version output version information and exit + +EOF + } + exit $exit_code; +} + +sub find_included_lib_files ($) +{ + my ($file) = @_; + + # Special cases... + my %special_non_dup = ( 'fnmatch_loop.c' => 1, + 'regex.c' => 1, 'at-func.c' => 1, + 'vasnprintf.c' => 1 + ); + my %dup_include_ok; + $dup_include_ok{'vasnprintf.c'}{'isnand-nolibm.h'} = 1; + $dup_include_ok{'vasnprintf.c'}{'isnanl-nolibm.h'} = 1; + $dup_include_ok{'vasnprintf.c'}{'fpucw.h'} = 1; + $dup_include_ok{'gen-uni-tables.c'}{'3level.h'} = 1; + $dup_include_ok{'csharpexec.c'}{'classpath.c'} = 1; + $dup_include_ok{'csharpexec.c'}{'classpath.h'} = 1; + + my %inc; + open FH, '<', $file + or die "$ME: can't open `$file' for reading: $!\n"; + + while (defined (my $line = <FH>)) + { + # Ignore test-driver code at end of file. + $line =~ m!^\#if(def)? TEST_! + and last; + + $line =~ m!^\s*\#\s*include\s+"! + or next; + $line =~ s///; + chomp $line; + $line =~ s/".*//; + exists $inc{$line} && ! exists $special_non_dup{$line} + && ! exists $dup_include_ok{basename $file}{$line} + and warn "$ME: $file: duplicate inclusion of $line\n"; + + $inc{$line} = 1; + } + close FH; + + return \%inc; +} + +my %exempt_header = + ( + # Exempt headers like unlocked-io.h that are `#include'd + # but not necessarily used. + 'unlocked-io.h' => 1, + + # Give gettext.h a free pass only when included from lib/error.c, + # since we've made that exception solely to make the error + # module easier to use -- at RMS's request. + 'lib/error.c:gettext.h' => 1, + + # The full-read module shares code with the full-write module. + 'lib/full-write.c:full-read.h' => 1, + + # The safe-write module shares code with the safe-read module. + 'lib/safe-read.c:safe-write.h' => 1, + + # The use of obstack.h in the hash module is conditional, off by default. + 'lib/hash.c:obstack.h' => 1, + + # C files in the gc module have conditional includes. + 'lib/gc-gnulib.c:des.h' => 1, + 'lib/gc-gnulib.c:arcfour.h' => 1, + 'lib/gc-gnulib.c:arctwo.h' => 1, + 'lib/gc-gnulib.c:md2.h' => 1, + 'lib/gc-gnulib.c:md4.h' => 1, + 'lib/gc-gnulib.c:md5.h' => 1, + 'lib/gc-gnulib.c:rijndael.h' => 1, + 'lib/gc-gnulib.c:sha1.h' => 1, + 'lib/gc-gnulib.c:rijndael-api-fst.h' => 1, + 'lib/gc-gnulib.c:hmac.h' => 1, + 'lib/gc-libgcrypt.c:md2.h' => 1, + ); + +sub check_module ($) +{ + my @m = @_; + + my %file; + my %module_all_files; + my %dep; + my %seen_module; + + while (@m) + { + my $m = pop @m; + # warn "M: $m\n"; + exists $seen_module{$m} + and next; + $seen_module{$m} = 1; + my ($file, $dep) = parse_module_file $m; + push @m, keys %$dep; + foreach my $f (keys %$file) + { + $module_all_files{$f} = 1; + } + } + + my @t = sort keys %module_all_files; + # warn "ALL files: @t\n"; + + # Derive from %module_all_files (by parsing the .c and .h files therein), + # the list of all #include'd files that reside in lib/. + foreach my $f (keys %module_all_files) + { + $f =~ /\.[ch]$/ + or next; + # FIXME: this is too naive + my $inc = find_included_lib_files "../$f"; + foreach my $i (sort keys %$inc) + { + my $lib_file = "lib/$i"; + exists $exempt_header{"$f:$i"} + || exists $exempt_header{$i} + and next; + !exists $module_all_files{$lib_file} && -f "../lib/$i" + and warn "$f: $i is `#include'd, but not " + . "listed in module's Files: section\n"; + } + #my @t = sort keys %$inc; + #print "** $f: @t\n"; + } +} + +{ + GetOptions + ( + help => sub { usage 0 }, + version => sub { print "$ME version $VERSION\n$COPYRIGHT_NOTICE"; exit }, + ) or usage 1; + + @ARGV < 1 + and (warn "$ME: missing FILE argument\n"), usage 1; + + foreach my $module (@ARGV) + { + check_module $module; + } + + exit 0; +} |