|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
use strict; |
|
|
use Getopt::Long; |
|
|
use File::Basename; |
|
|
|
|
|
|
|
|
my $COPYRIGHT_NOTICE = "Copyright (C) 2012 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 <https://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 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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) = @_; |
|
|
|
|
|
|
|
|
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>)) |
|
|
{ |
|
|
|
|
|
$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 = |
|
|
( |
|
|
|
|
|
|
|
|
'unlocked-io.h' => 1, |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
'lib/error.c:gettext.h' => 1, |
|
|
|
|
|
|
|
|
'lib/full-write.c:full-read.h' => 1, |
|
|
|
|
|
|
|
|
'lib/safe-read.c:safe-write.h' => 1, |
|
|
|
|
|
|
|
|
'lib/hash.c:obstack.h' => 1, |
|
|
|
|
|
|
|
|
'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; |
|
|
|
|
|
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; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
foreach my $f (keys %module_all_files) |
|
|
{ |
|
|
$f =~ /\.[ch]$/ |
|
|
or next; |
|
|
|
|
|
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"; |
|
|
} |
|
|
|
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
{ |
|
|
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; |
|
|
} |
|
|
|