;# $Id$ ;# ;# Copyright (c) 1991-1997, 2004-2006, Raphael Manfredi ;# ;# You may redistribute only under the terms of the Artistic Licence, ;# as specified in the README file that comes with the distribution. ;# You may reuse parts of this distribution only within the terms of ;# that same Artistic Licence; a copy of which may be found at the root ;# of the source tree for dist 4.0. ;# ;# $Log: lint.pl,v $ ;# Revision 3.0.1.10 1997/02/28 16:31:53 ram ;# patch61: added support for ?F: lines to monitor file usage ;# patch61: now honours "create" and "empty" lint directives ;# ;# Revision 3.0.1.9 1995/09/25 09:19:15 ram ;# patch59: new ?Y: directive to change unit layout ;# ;# Revision 3.0.1.8 1995/07/25 14:19:47 ram ;# patch56: will now check : comments line for potential danger ;# ;# Revision 3.0.1.7 1994/10/29 16:36:14 ram ;# patch36: now extensively checks created files thanks to new ?F: lines ;# ;# Revision 3.0.1.6 1994/05/13 15:29:09 ram ;# patch27: now understands macro definitions in ?H: lines ;# ;# Revision 3.0.1.5 1994/05/06 15:27:48 ram ;# patch23: now warns for units ending with non-blank line ;# patch23: warn for units where last line is not new-line terminated ;# ;# Revision 3.0.1.4 1994/01/24 14:28:40 ram ;# patch16: now knows about "internal use only" variables on ?MAKE: lines ;# patch16: now suppress "read-only var set" message when change hint ;# ;# Revision 3.0.1.3 1993/11/10 17:39:39 ram ;# patch14: now spots stale ?M: dependencies ;# ;# Revision 3.0.1.2 1993/10/16 13:55:26 ram ;# patch12: now checks ?M: lines also ;# ;# Revision 3.0.1.1 1993/08/25 14:03:40 ram ;# patch6: now correctly signals conditional dependencies with no default ;# ;# Revision 3.0 1993/08/18 12:10:25 ram ;# Baseline for dist 3.0 netwide release. ;# ;# The list of all available units is held in @ARGV. We shall parse them and ;# extract the dependencies. A lot of global data structures are filled in ;# during this phase. ;# # Initialize the extraction process by setting some variables. # We return a string to be eval'ed to do more customized initializations. sub init_extraction { $c_symbol = ''; # Current symbol seen in ?C: lines $s_symbol = ''; # Current symbol seen in ?S: lines $m_symbol = ''; # Current symbol seen in ?M: lines $h_section = 0; # 0 = no ?H: yet, 1 = in ?H:, 2 = ?H:. seen $h_section_warned = 0; # Whether we warned about terminated ?H: section $heredoc = ''; # Last "here" document symbol seen $heredoc_nosubst = 0; # True for <<'EOM' here docs $heredoc_line = 0; # Line were last "here" document started $last_interpreted = 0; # True when last line was an '@' one $past_first_line = 0; # True when first body line was already seen $wiped_unit = 0; # True if unit will be "wiped" for macro subst %csym = (); # C symbols described %ssym = (); # Shell symbols described %hcsym = (); # C symbols used by ?H: lines %hssym = (); # Shell symbols used by ?H: lines %msym = (); # Magic symbols defined by ?M: lines %mdep = (); # C symbol dependencies introduced by ?M: %symset = (); # Records all the shell symbol set %symused = (); # Records all the shell symbol used %tempseen = (); # Temporary shell variable seen %fileseen = (); # Produced files seen %fileused = (); # Files used, by unit (private UU files) %filemisused = (); # Files not used as ./file or ...UU/file %filetmp = (); # Local temporary files in ?F: directives %filesetin = (); # Lists units defining a temporary file %filecreated = (); # Records files created in this unit %prodfile = (); # Unit where a given file is said to be created %defseen = (); # Symbol defintions claimed %lintset = (); # Symbols declared set by a ?LINT: line %lintsdesc = (); # Symbols declared described by a ?LINT: line %lintcdesc = (); # Symbols declared described by a ?LINT: line %lintseen = (); # Symbols declared known by a ?LINT: line %lintchange = (); # Symbols declared changed by a ?LINT: line %lintuse = (); # Symbols declared used by unit %lintextern = (); # Symbols known to be externally defined %lintcreated = (); # Files declared as created by a ?LINT: line %linthere = (); # Unclosed here document from ?LINT: line %lintnothere = (); # False here document names, from ?LINT: line %lintfused = (); # Records files markedas used in ?LINT: line %lintchange_used = (); # Tracks symbols for which %lintchange was used %lintuse_used = (); # Tracks symbols for which %lintuse was used %lintseen_used = (); # Tracks symbols for which %lintseen was used %lintcdesc_used = (); # Tracks symbols for which %lintcdesc was used %lintsdesc_used = (); # Tracks symbols for which %lintsdesc was used %lintset_used = (); # Tracks symbols for which %lintset was used %lintnocomment = (); # Signals it's OK for unit to lack a : comment %condsym = (); # Records all the conditional symbols %condseen = (); # Records conditional dependencies %depseen = (); # Records full dependencies %shvisible = (); # Records units making a symbol visible %shspecial = (); # Records special units listed as wanted %shdepend = (); # Records units listed in one's dependency list %shmaster = (); # List of units defining a shell symbol %cmaster = (); # List of units defining a C symbol %symdep = (); # Records units where symbol is a dependency @make = (); # Records make dependency lines $body = 'p_body'; # Procedure to handle body $ending = 'p_end'; # Called at the end of each unit @wiping = # The keywords we recognize for "wiped" units qw( PACKAGENAME MAINTLOC VERSION PATCHLEVEL REVISION DATE BASEREV ); } # End the extraction process sub end_extraction { } # Process the command line of ?MAKE: lines sub p_make_command { local ($_) = @_; my $where = "\"$file\", line $. (?MAKE:)"; unless (s/^\t+//) { warn "$where: command line must start with a leading TAB character.\n"; s/^\s+//; # Remove spaces and continue } return unless s/^-?pick\b//; # Validate the special "pick" make command, processed internally # by metaconfig. my %valid = map { $_ => 1 } qw( add add.Config_sh add.Null c_h_weed cm_h_weed close.Config_sh prepend weed wipe ); my $cmd; $cmd = $1 if s/^\s+(\S+)//; unless (defined $cmd) { warn "$where: pick needs a command argument.\n"; return; } $wiped_unit++ if $cmd eq 'wipe'; warn "$where: unknown pick command '$cmd'.\n" unless $valid{$cmd}; s/^\s+//; unless (s/^\$\@//) { warn "$where: third pick argument must be \$\@\n"; return; } s/^\s+//; my $target; $target = $1 if s/^(\S+)//; unless (defined $target) { warn "$where: fourth pick argument is missing\n"; return; } return if $target =~ m|^\./|; warn "$where: weird fourth argument '$target' to pick.\n" unless $target =~ /^\w+$/; warn "$where: fourth pick argument should probably be the %< macro.\n" unless $target eq $unit; } # Process the ?MAKE: line sub p_make { local($_) = @_; local(@ary); # Locally defined symbols local(@dep); # Dependencies local($where) = "\"$file\", line $. (?MAKE:)"; unless (/^[\w+ ]*:/) { &p_make_command; return; # We only want the main dependency rule } warn "$where: ignoring duplicate dependency listing line.\n" if $makeseen{$unit}++; return if $makeseen{$unit} > 1; # Reset those once for every unit # (assuming there is only one depend line) $h_section = 0; # 0 = no ?H: yet, 1 = in ?H:, 2 = ?H:. seen $h_section_warned = 0; # Whether we warned about terminated ?H: section $wiped_unit = 0; # Whether macros like " will be wiped undef %condseen; undef %depseen; undef %defseen; undef %tempseen; undef %symset; undef %symused; undef %csym; undef %ssym; undef %hcsym; undef %hssym; undef %lintuse; undef %lintuse_used; undef %lintseen; undef %lintchange; undef %lintchange_used; undef %lintextern; undef %lintcreated; undef %fileseen; undef %lintseen_used; undef %filetmp; undef %filecreated; undef %linthere; undef %lintnothere; undef %lintfused; undef %lintsdesc; undef %lintsdesc_used; undef %lintcdesc; undef %lintcdesc_used; undef %lintset; undef %lintset_used; s|^\s*||; # Remove leading spaces chop; s/:(.*)//; @dep = split(' ', $1); # Dependencies @ary = split(' '); # Locally defined symbols local($nowarn); # True when +Special is seen foreach $sym (@ary) { # Ignore "internal use only" symbols as far as metalint goes. # Actually, we record the presence of a '+' in front of a special # unit name and use that as a hint to suppress the presence of that # special unit in the defined symbol section. $nowarn = ($sym =~ s/^\+//); # We record for each shell symbol the list of units which claim to make # it, so as to report duplicates. if ($sym =~ /^[_a-z]/ || $Except{$sym}) { $shmaster{"\$$sym"} .= "$unit "; ++$defseen{$sym}; } else { warn "$where: special unit '$sym' should not be listed as made.\n" unless $sym eq $unit || $nowarn; } } # Record dependencies for later perusal push(@make, join(' ', @ary) . ':' . join(' ', @dep)); foreach $sym (@dep) { if ($sym =~ /^\+[_A-Za-z]/) { $sym =~ s|^\+||; ++$condseen{$sym}; # Conditional symbol wanted ++$condsym{$sym}; # %condsym has a greater lifetime } else { ++$depseen{$sym}; # Full dependency } # Each 'wanted' special unit (i.e. one starting with a capital letter) # is remembered, so as to prevent exported symbols from being reported # as "undefined". For instance, Myread exports $dflt, $ans and $rp. $shspecial{$unit} .= "$sym " if substr($sym, 0, 1) =~ /^[A-Z]/; # Record all known dependencies (special or not) for this unit $shdepend{$unit} .= "$sym "; # Remember where wanted symbol is defined, so that we can report # stale dependencies later on (i.e. dependencies which refer to non- # existent symbols). $symdep{$sym} .= "$unit "; # This symbol is wanted here } # Make sure we do not want a symbol twice, nor do we want it once as a full # dependency and once as a conditional dependency. foreach $sym (@dep) { if ($sym =~ /^\+[_A-Za-z]/) { $sym =~ s|^\+||; warn "$where: '+$sym' is listed $condseen{$sym} times.\n" if $condseen{$sym} > 1; $condseen{$sym} = 1 if $condseen{$sym}; # Avoid multiple messages } else { warn "$where: '$sym' is listed $depseen{$sym} times.\n" if $depseen{$sym} > 1; $depseen{$sym} = 1 if $depseen{$sym}; # Avoid multiple messages } warn "$where: '$sym' listed as both conditional and full dependency.\n" if $condseen{$sym} && $depseen{$sym}; } # Make sure every unit "inherits" from the symbols exported by 'Init'. $shspecial{$unit} .= 'Init ' unless $shspecial{$unit} =~ /Init\b/; } # Process the ?O: line sub p_obsolete { local($_) = @_; chop; $Obsolete{"$unit.U"} = $_; # Message to print if unit is used } # Process the ?S: lines sub p_shell { local($_) = @_; local($where) = "\"$file\", line $. (?S:)"; warn "$where: directive should come after ?MAKE declarations.\n" unless $makeseen{$unit}; if (/^(\w+)\s*(\(.*\))*\s*:/) { &check_last_declaration; $s_symbol = $1; print " ?S: $s_symbol\n" if $opt_d; # Make sure we do not define symbol twice and that the symbol is indeed # listed in the ?MAKE: line. warn "$where: duplicate description for variable '\$$s_symbol'.\n" if $ssym{$s_symbol}++; unless ($defseen{$s_symbol}) { warn "$where: variable '\$$s_symbol' is not listed " . "on ?MAKE: line.\n" unless $lintseen{$s_symbol}; $lintseen_used{$s_symbol}++ if $lintseen{$s_symbol}; } # Deal with obsolete symbol list (enclosed between parenthesis) &record_obsolete("\$$_") if /\(/; } else { unless ($s_symbol) { warn "$where: syntax error in ?S: construct.\n"; return; } } m|^\.\s*$| && ($s_symbol = ''); # End of comment } # Process the ?C: lines sub p_c { local($_) = @_; local($where) = "\"$file\", line $. (?C:)"; warn "$where: directive should come after ?MAKE declarations.\n" unless $makeseen{$unit}; # The previous ?H: section, if present, must have been closed if ($h_section && $h_section != 2) { warn "$where: unclosed ?H: section.\n"; } $h_section = 0; if (s/^(\w+)\s*~\s*(\S+)\s*(.*):/$1 $3:/) { &check_last_declaration; $c_symbol = $2; # Alias for definition in config.h # Record symbol definition for further duplicate spotting $cmaster{$1} .= "$unit " unless $csym{$1}; print " ?C: $1 ~ $c_symbol\n" if $opt_d; # Make sure we do not define symbol twice warn "$where: duplicate description for symbol '$1'.\n" if $csym{$1}++; # Deal with obsolete symbol list (enclosed between parenthesis) &record_obsolete("$_") if /\(/; } elsif (/^(\w+)\s*(\(.*\))*\s*:/) { &check_last_declaration; $c_symbol = $1; # Record symbol definition for further duplicate spotting $cmaster{$c_symbol} .= "$unit " unless $csym{$c_symbol}; print " ?C: $c_symbol\n" if $opt_d; # Make sure we do not define symbol twice warn "$where: duplicate description for symbol '$c_symbol'.\n" if $csym{$c_symbol}++; # Deal with obsolete symbol list (enclosed between parenthesis) &record_obsolete("$_") if /\(/; } else { unless ($c_symbol) { warn "$where: syntax error in ?C: construct.\n"; return; } } s|^(\w+)|?$c_symbol:/* $1| || # Start of comment (s|^\.\s*$|?$c_symbol: */\n| && ($c_symbol = '', 1)) || # End of comment s|^(.*)|?$c_symbol: *$1|; # Middle of comment } # Process the ?H: lines sub p_config { local($_) = @_; local($where) = "\"$file\", line $. (?H)" unless $where; warn "$where: directive should come after ?MAKE declarations.\n" unless $makeseen{$unit}; unless ($h_section){ # Entering ?H: section $h_section = 1; $h_section_warned = 0; } if ($h_section == 2) { warn "$where: section was already terminated by '?H:.'.\n" unless $h_section_warned++; return; } if ($_ eq ".\n") { $h_section = 2; # Marks terminated ?H: section return; } (my $constraint) = m/^\?(\w+):/; s/^\?\w+://; # Remove leading '?var:' constraint if (m|^#\$(\w+)\s+(\w+).*\$(\w+)|) { # Case: #$d_var VAR "$var" warn "$where: symbol '$2' was already defined.\n" if $hcsym{$2}++; &check_definition("$1"); &check_definition("$3"); } elsif (m|^#define\s+(\w+)\((.*)\)\s+\$(\w+)|) { # Case: #define VAR(x) $var warn "$where: symbol '$1' was already defined.\n" if $hcsym{$1}++; &check_definition("$3"); } elsif (m|^#\$define\s+(\w+)|) { # Case: #$define VAR warn "$where: symbol '$1' was already defined.\n" if $hcsym{$1}++; } elsif (m|^#\$(\w+)\s+(\w+)|) { # Case: #$d_var VAR warn "$where: symbol '$2' was already defined.\n" if $hcsym{$2}++; &check_definition("$1"); } elsif (m|^#define\s+(\w+).*\$(\w+)|) { # Case: #define VAR "$var" warn "$where: symbol '$1' was already defined.\n" if $hcsym{$1}++; &check_definition("$2"); } elsif (m|^#define\s+(\w+)|) { # Case: #define VAR $hcsym{$1}++; # Multiple occurrences may be legitimate } else { if (/^#/) { warn "$where: uncommon cpp line should be protected with '?%<:'.\n" if $constraint eq ''; } elsif (!/^\@(if|elsif|else|end)\b/) { warn "$where: line should not be listed here but in '?C:'.\n"; } } # Ensure the constraint is either %< (unit base name) or a known symbol. if ($constraint ne '' && $constraint ne $unit) { warn "$where: constraint '$constraint' is an unknown symbol.\n" unless $csym{$constraint} || $ssym{$constraint}; } } # Process the ?M: lines sub p_magic { local($_) = @_; local($where) = "\"$file\", line $. (?M)"; warn "$where: directive should come after ?MAKE declarations.\n" unless $makeseen{$unit}; if (/^(\w+):\s*([\w\s]*)\n$/) { &check_last_declaration; $m_symbol = $1; $msym{$1} = "$unit"; # p_wanted ensure we do not define symbol twice $mdep{$1} = $2; # Save C symbol dependencies &p_wanted("$unit:$m_symbol"); } else { unless ($m_symbol) { warn "$where: syntax error in ?M: construct.\n"; return; } } m|^\.\s*$| && ($m_symbol = ''); # End of comment } # Process the ?INIT: lines sub p_init { local($_) = @_; local($where) = "\"$file\", line $. (?INIT)"; warn "$where: directive should come after ?MAKE declarations.\n" unless $makeseen{$unit}; &p_body($_, 1); # Pass it along as a body line (leading ?INIT: removed) } # Process the ?D: lines sub p_default { local($_) = @_; local($where) = "\"$file\", line $. (?D)"; warn "$where: directive should come after ?MAKE declarations.\n" unless $makeseen{$unit}; local($sym) = /^(\w+)=/; $hasdefault{$sym}++; unless ($defseen{$sym}) { warn "$where: variable '\$$sym' is not listed " . "on ?MAKE: line.\n" unless $lintseen{$sym}; $lintseen_used{$sym}++ if $lintseen{$sym}; } s/^\w+=//; # So that p_body does not consider variable as being set &p_body($_, 1); # Pass it along as a body line (leading ?D: + var removed) } # Process the ?V: lines sub p_visible { local($where) = "\"$file\", line $. (?V)"; warn "$where: directive should come after ?MAKE declarations.\n" unless $makeseen{$unit}; # A visible symbol can freely be manipulated by any unit which includes the # current unit in its dependencies. Symbols before ':' may be only used for # reading while symbols after ':' may be used for both reading and writing. # The array %shvisible records symbols as keys. Read-only symbols have a # leading '$' while read-write symbols are recorded as-is. unless (substr($unit, 0, 1) =~ /^[A-Z]/) { warn "$where: visible declaration in non-special unit ignored.\n"; return; } local($read_only) = $_[0] =~ /^([^:]*):?/; local($read_write) = $_[0] =~ /:(.*)/; local(@rsym) = split(' ', $read_only); local(@rwsym) = split(' ', $read_write); local($w); foreach (@rsym) { # Read only symbols warn "$where: wanted variable '\$$_' made visible.\n" if &wanted($_); warn "$where: defined variable '\$$_' made visible.\n" if &defined($_) && !$lintseen{$_}; $w = $shvisible{"\$$_"}; warn "$where: variable '\$$_' already made visible by unit $w.\n" if $w; $w = $shvisible{$_}; warn "$where: variable '\$$_' already read-write visible in $w.\n" if $w; $shvisible{"\$$_"} = $unit unless $w; } foreach (@rwsym) { # Read/write symbols warn "$where: wanted variable '\$$_' made visible.\n" if &wanted($_); warn "$where: defined variable '\$$_' made visible.\n" if &defined($_) && !$lintseen{$_}; $w = $shvisible{$_}; warn "$where: variable '\$$_' already made visible by unit $w.\n" if $w; $w = $shvisible{"\$$_"}; warn "$where: variable '\$$_' already read-only visible in $w.\n" if $w; $shvisible{$_} = $unit unless $w; } } # Process the ?W: lines sub p_wanted { local($where) = "\"$file\", line $. (?W)" unless $where; warn "$where: directive should come after ?MAKE declarations.\n" unless $makeseen{$unit}; # Somehow, we should check that none of the symbols to activate are stale # ones, i.e. they all finally resolve to some known target -- FIXME local($active) = $_[0] =~ /^([^:]*):/; # Symbols to activate local($look_symbols) = $_[0] =~ /:(.*)/; # When those are used local(@symbols) = split(' ', $look_symbols); # A "?W:symbol" line asks metaconfig to define 'symbol' in the wanted file # as a C target iff that word is found within the sources. This is mainly # intended for the built-in interpreter to check for definedness. local($w); foreach (@symbols) { warn "$where: variable '\$$_' already wanted.\n" if &wanted($_); warn "$where: variable '\$$_' also locally defined.\n" if &defined($_); $w = $cwanted{$_}; if ($msym{$_} ne '') { warn "$where: symbol '$_' already listed on a ?M: line in '$w'.\n" if $w; } else { warn "$where: variable '\$$_' already listed on a ?W: line in '$w'.\n" if $w; } $cwanted{$_} = $unit unless $w; } } # Process the ?Y: lines sub p_layout { local($where) = "\"$file\", line $. (?Y)"; warn "$where: directive should come after ?MAKE declarations.\n" unless $makeseen{$unit}; local($_) = @_; chop; s/^\s+//; tr/A-Z/a-z/; # Layouts are record in lowercase warn "$where: unknown layout directive '$_'.\n" unless defined $Lcmp{$_}; } # Process the ?P: lines sub p_public { # FIXME } # Process the ?L: lines sub p_library { # There should not be any '-l' in front of the library name # FIXME } # Process the ?I: lines sub p_include { # FIXME } # Process the ?T: lines sub p_temp { local($where) = "\"$file\", line $. (?T:)"; warn "$where: directive should come after ?MAKE declarations.\n" unless $makeseen{$unit}; local($_) = @_; local(@sym) = split(' ', $_); foreach $sym (@sym) { warn "$where: temporary symbol '\$$sym' multiply declared.\n" if $tempseen{$sym}++ == 1; $tempmaster{$sym} .= "$unit " if $tempseen{$sym} == 1; } } # Process the ?F: lines sub p_file { local($where) = "\"$file\", line $. (?F:)"; warn "$where: directive should come after ?MAKE declarations.\n" unless $makeseen{$unit}; local($_) = @_; local(@files) = split(' ', $_); local($uufile); # Name of file produced in the UU directory local($tmpfile); # Name of a temporary file # We care only about UU files, i.e. files produced in the UU directory # and which are identified by the convention ./filename. Files !filename # are not produced, i.e. they are temporary or externally provided. # The %prodfile table records all the files produced, so we may detect # inconsistencies between units, while %filemaster records the (first) unit # defining a given UU file to make sure that (special) unit is named in the # dependency line when that UU file if used. Duplicates will be caught in # the sanity check phase thanks to %prodfile. # Temporary files are recorded in %filesetin, so that we may later compare # the list with the UU files to detect possible overwrites. my $is_special = substr($unit, 0, 1) =~ /^[A-Z]/; foreach $file (@files) { warn "$where: produced file '$file' multiply declared.\n" if $fileseen{$file}++ == 1; if (($tmpfile = $file) =~ s/^!//) { $filetmp{$tmpfile} = 'x '; $filesetin{$tmpfile} .= "$unit " if $fileseen{$file} == 1; next; # Is not a UU file for sure, so skip } $prodfile{$file} .= "$unit " if $fileseen{$file} == 1; ($uufile = $file) =~ s|^\./(\S+)$|$1|; next if $file eq $uufile; # Don't care about non-UU files unless ($is_special || $lintcreated{$uufile}) { warn "$where: UU file '$uufile' in non-special unit ignored.\n"; delete $lintcreated{$uufile}; # Detect spurious LINT next; } delete $lintcreated{$uufile} if !$is_special; # Detect spurious LINT $filemaster{$uufile} = $unit unless defined $filemaster{$uufile}; $filecreated{$uufile} = 'a'; # Will be automagically incremented } } # Process the ?LINT: lines sub p_lint { local($_) = @_; local(@sym); local($where) = "\"$file\", line $. (?LINT:)"; s/^\s+//; # Strip leading spaces unless ($makeseen{$unit}) { warn "$where: directive should come after ?MAKE declarations.\n" unless m/^empty/; } if (s/^set//) { # Listed variables are set @sym = split(' ', $_); # Spurious ones will be flagged foreach (@sym) { $lintset{$_}++; # Shell variable set } } elsif (s/^desc\w+//) { # Listed shell variables are described @sym = split(' ', $_); # Spurious ones will be flagged foreach (@sym) { $lintsdesc{$_}++; # Shell variable described } } elsif (s/^creat\w+//) { # Listed created files in regular units @sym = split(' ', $_); foreach (@sym) { $lintcreated{$_}++; # Persistent UU file created } } elsif (s/^known//) { # Listed C variables are described @sym = split(' ', $_); # Spurious ones will be flagged foreach (@sym) { $lintcdesc{$_}++; # C symbol described } } elsif (s/^change//) { # Shell variable ok to be changed @sym = split(' ', $_); # Spurious ones will be flagged foreach (@sym) { $lintchange{$_}++; # Do not complain if changed } } elsif (s/^extern//) { # Variables known to be externally defined @sym = split(' ', $_); foreach (@sym) { $lintextern{$_}++; # Do not complain if used in a ?H: line } } elsif (s/^usefile//) { # Files marked as being used @sym = split(' ', $_); foreach (@sym) { $lintfused{$_}++; } } elsif (s/^use//) { # Variables declared as used by unit @sym = split(' ', $_); # Spurious ones will be flagged foreach (@sym) { $lintuse{$_}++; # Do not complain if on ?MAKE and not used } } elsif (s/^def\w+//) { # Listed variables are defined @sym = split(' ', $_); # Spurious ones will be flagged foreach (@sym) { $lintseen{$_}++; # Shell variable defined in this unit } } elsif (m/^empty/) { # Empty unit file $lintempty{$unit}++; } elsif (m/^unclosed/) { # Unclosed here-documents @sym = split(' ', $_); foreach (@sym) { $linthere{$_}++; } } elsif (s/^nothere//) { # Not a here-document name @sym = split(' ', $_); foreach (@sym) { $lintnothere{$_}++; } } elsif (s/^nocomment//) { # OK if leading unit ': comment' missing $lintnocomment{$unit}++; } else { local($where) = "\"$file\", line $." unless $where; local($word) = /^(\w+)/; warn "$where: unknown LINT request '$word' ignored.\n"; } } # Process the body of the unit sub p_body { return unless $makeseen{$unit}; local($_, $special) = @_; local($where) = "\"$file\", line $." unless $where; # Ensure there is no control line in the body of the unit local($control) = /^\?([\w\-]+):/; local($known) = $control ? $Depend{$control} : ""; warn "$where: control sequence '?$control:' ignored within body.\n" if $known && !/^\?X:|^\?LINT:/; if (s/^\?LINT://) { # ?LINT directives allowed within body $_ .= &complete_line(FILE) if s/\\\s*$//; &p_lint($_); } return if $known; # First non-special line should be a ': description' line unless ($special || /^\?/ || /^@/) { warn "$where: first body line should be a general ': description'.\n" unless $past_first_line++ || $lintnocomment{$unit} || /^:\s+\w+/; } # Ensure ': comment' lines do not hold any meta-character # We assume ":)" introduces a case statement. if (/^\s*:/ && !/^\s*:\)/) { warn "$where: missing space after ':' to make it a comment.\n" unless /^\s*:\s/; s/\\.//g; # simplistic ignoring of "escaped" chars s/".*?"//g; s/'.*?'//g; if ($wiped_unit) { s/<\$\w+>//g; foreach my $wipe (@wiping) { s/<$wipe>//g; } } warn "$where: found unquoted meta-character $1 on comment line.\n" while s/([`()<>;&\{\}\|])//g; warn "$where: found dangling quote on ':' comment line.\n" if /['"]/; return; } # Ingnore interpreted lines and their continuations if ($last_interpreted) { return if /\\$/; # Still part of the interpreted line $last_interpreted = 0; # End of interpreted lines return; # This line was the last interpreted } # Look for interpreted lines and ignore them if (/^@/) { $last_interpreted = /\\$/; # Set flag if line is continued return; # And skip this line } # Detect ending of "here" documents if ($heredoc ne '' && $_ eq "$heredoc\n") { $heredoc = ''; # Close here-document $heredoc_nosubst = 0; return; } # Detect beginning of "here" document my $began_here = 0; if ($heredoc eq '') { if (/<<\s*''/) { # Discourage it, because we're not processing those... warn "$where: empty here-document name discouraged.\n"; } elsif (/<<\s*'([^']+)'/ && !$lintnothere{$1}) { $heredoc = $1; $heredoc_nosubst = 1; $began_here++; } elsif (/<<\s*(\S+)/ && !$lintnothere{$1}) { $heredoc = $1; $began_here++; } # Continue, as we need to look for possible ">file" on the same line # as a possible here document, as in "cat <file". } else { return if $heredoc_nosubst; # Completely opaque to interpretation } $heredoc_line = $. if $began_here; # If we've just entered a here document and we're generating a file # that is exported by the unit, then we need to monitor the variables # used to make sure there's no missing dependency. $heredoc_nosubst = 0 if $began_here && />>?\s*(\S+)/ && $filemaster{$1} eq $unit; # From now on, do all substitutes with ':' since it would be dangerous # to remove things plain and simple. It could yields false matches # afterwards... my $check_vars = 1; $check_vars = 0 if $heredoc_nosubst && !$began_here; # Record any attempt made to set a shell variable local($sym); while ($check_vars && s/(\W?)(\w+)=/$1:/) { my $before = $1; $sym = $2; next unless $before eq '' || $before =~ /["'` \t]/; next if $sym =~ /^\d+/; # Ignore $1 and friends $symset{$sym}++; # Shell variable set # Not part of a $cc -DWHATEVER line and not made nor temporary unless ($sym =~ /^D/ || &defined($sym)) { if (&wanted($sym)) { warn "$where: variable '\$$sym' is changed.\n" unless $lintchange{$sym}; $lintchange_used{$sym}++ if $lintchange{$sym}; } else { # Record that the variable is set but not listed locally. if ($shset{$unit} !~ /\b$sym\b/) { $shset{$unit} .= "$sym " unless $lintchange{$sym}; $lintchange_used{$sym}++ if $lintchange{$sym}; } } } } # Now look at the shell variables used: can be $var or ${var} or ${var: local($var); local($line) = $_; while ($check_vars && s/\$\{?(\w+)[\}:]?/$1/) { $var = $1; next if $var =~ /^\d+/; # Ignore $1 and friends # Record variable as undeclared but do not issue a message right now. # That variable could be exported via ?V: (as $dflt in Myread) or be # defined by a special unit (like $inlibc by unit Inlibc). $shunknown{$unit} .= "$var " unless $lintextern{$var} || &declared($var) || $shunknown{$unit} =~ /\b$var\b/; $shused{$unit} .= "\$$var " unless $shused{$unit} =~ /\$$var\b/; } local($file); if ($heredoc ne '' && !$began_here) { # Still in here-document # Just look for included files from C programs expected to be local # in case they missed the special unit that produces these files. if (s!#(\s*)include(\s+)"([\w.]+)"!!) { $file = $3; $fileused{$unit} .= "$file " unless $filetmp{$file} || $fileused{$unit} =~ /\b$file\b/; } return; } # Now look at private files used by the unit (./file or ..../UU/file) # We look at things like '. ./myread' and `./loc ...` as well as "< file" $_ = $line; s/<\S+?>//g; # would set-off our >file or >file while (s!>>?\s*([^\$/`\s;]+)\s*!: !) { $file = $1; next if $file =~ /&\d+/; # skip >&4 and friends $filecreated{$file}++; } # Look for mentions of known temporary files to avoid complaining # that they were not used. while (s!\s+(\S+)!!) { $file = $1; $filetmp{$file} .= ' used' if defined $filetmp{$file} && $filetmp{$file} !~ /\bused/; } } # Called at the end of each unit sub p_end { local($last) = @_; # Last processed line local($where) = "\"$file\""; # The ?H: section, if present, must have been closed if ($h_section && $h_section != 2) { warn "$where: unclosed ?H: section.\n"; } $h_section = 0; # For next unit, which may be empty # All opened here-documents must be closed. if ($heredoc ne '') { my $q = $heredoc_nosubst ? "'" : ""; warn "$where: unclosed here-document $q$heredoc$q " . "started line $heredoc_line.\n" unless $linthere{$heredoc}; } # Reinitialize for next unit. $heredoc = ''; $heredoc_nosubst = 0; $past_first_line = 0; $last_interpreted = 0; unless ($makeseen{$unit}) { warn "$where: no ?MAKE: line describing dependencies.\n" unless $lintempty{$unit}; return; } # Each unit should end with a blank line. Unfortunately, some units # may also end with an '@end' request and have the blank line above it. # Currently, we do not have enough information to correctly diagnose # whether it is valid or not so just skip it. # Same thing for U/Obsol_sh.U which ends with a shell comment. warn "$where: not ending with a blank line.\n" unless $last =~ /^\s*$/ || $last =~ /^\@end/ || $last =~ /^#|\?/; # For EMACS users. It would be fatal to the Configure script... warn "$where: last line not ending with a new-line character.\n" unless $last =~ /\n$/; # Make sure every shell symbol described in ?MAKE had a description foreach $sym (sort keys %defseen) { unless ($ssym{$sym}) { warn "$where: symbol '\$$sym' was not described.\n" unless $lintsdesc{$sym}; $lintsdesc_used{$sym}++ if $lintsdesc{$sym}; } } # Ensure all the C symbols defined by ?H: lines have a description foreach $sym (sort keys %hcsym) { unless ($csym{$sym}) { warn "$where: C symbol '$sym' was not described.\n" unless $lintcdesc{$sym}; $lintcdesc_used{$sym}++ if $lintcdesc{$sym}; } } # Ensure all the C symbols described by ?C: lines are defined in ?H: foreach $sym (sort keys %csym) { warn "$where: C symbol '$sym' was not defined by any ?H: line.\n" unless $hcsym{$sym}; } # Make sure each defined symbol was set, unless it starts with an # upper-case letter in which case it is not a "true" shell symbol. # I don't care about the special symbols defined in %Except as I know # they are handled correctly. foreach $sym (sort keys %defseen) { unless ($symset{$sym} || substr($sym, 0, 1) =~ /^[A-Z]/) { warn "$where: variable '\$$sym' should have been set.\n" unless $lintset{$sym}; $lintset_used{$sym}++ if $lintset{$sym}; } } # Make sure every non-special unit declared as wanted is indeed needed foreach $sym (sort keys %depseen) { if ($shused{$unit} !~ /\$$sym\b/ && substr($sym, 0, 1) !~ /^[A-Z]/) { warn "$where: unused dependency variable '\$$sym'.\n" unless $lintchange{$sym} || $lintuse{$sym}; $lintchange_used{$sym}++ if $lintchange{$sym}; $lintuse_used{$sym}++ if $lintuse{$sym}; } } # Idem for conditionally wanted symbols foreach $sym (sort keys %condseen) { if ($shused{$unit} !~ /\$$sym\b/ && substr($sym, 0, 1) !~ /^[A-Z]/) { warn "$where: unused conditional variable '\$$sym'.\n" unless $lintchange{$sym} || $lintuse{$sym}; $lintchange_used{$sym}++ if $lintchange{$sym}; $lintuse_used{$sym}++ if $lintuse{$sym}; } } # Idem for temporary symbols foreach $sym (sort keys %tempseen) { if ($shused{$unit} !~ /\$$sym\b/ && !$symset{$sym}) { warn "$where: unused temporary variable '\$$sym'.\n" unless $lintuse{$sym}; $lintuse_used{$sym}++ if $lintuse{$sym}; } } # Idem for local files foreach $file (sort keys %filetmp) { warn "$where: mis-used temporary file '$file'.\n" if $filetmp{$file} =~ /\bmisused/; warn "$where: unused temporary file '$file'.\n" unless $lintfused{$file} || $filetmp{$file} =~ /\bused/ || $filetmp{$file} =~ /\bmisused/; } # Make sure each private file listed as created on ?F: is really created. # When found, a private UU file is entered in the %filecreated array # with value 'a'. Each time a file creation occurs in the unit, an # increment is done on that value. Since 'a'++ -> 'b', a numeric value # in %filecreated means a non-local file, which is skipped. An 'a' means # the file was not created... local($value); foreach $file (sort keys %filecreated) { $value = $filecreated{$file}; next if $value > 0; # Skip non UU-files. warn "$where: file '$file' was not created.\n" if $value eq 'a'; } # Check whether some of the LINT directives were useful foreach my $sym (sort keys %lintcreated) { warn "$where: spurious 'LINT create $sym' directive.\n"; } foreach my $sym (sort keys %lintuse) { warn "$where: spurious 'LINT use $sym' directive.\n" unless $lintuse_used{$sym}; } foreach my $sym (sort keys %lintchange) { warn "$where: spurious 'LINT change $sym' directive.\n" unless $lintchange_used{$sym}; } foreach my $sym (sort keys %lintseen) { warn "$where: spurious 'LINT define $sym' directive.\n" unless $lintseen_used{$sym}; } foreach my $sym (sort keys %lintsdesc) { warn "$where: spurious 'LINT describe $sym' directive.\n" unless $lintsdesc_used{$sym}; } foreach my $sym (sort keys %lintcdesc) { warn "$where: spurious 'LINT known $sym' directive.\n" unless $lintcdesc_used{$sym}; } foreach my $sym (sort keys %lintset) { warn "$where: spurious 'LINT set $sym' directive.\n" unless $lintset_used{$sym}; } } # An unknown control line sequence was found (held in $proc) sub p_unknown { warn "\"$file\", line $.: unknown control sequence '?$proc:'.\n"; } # Run sanity checks, to make sure every conditional symbol has a suitable # default value. Also ensure every symbol was defined once. sub sanity_checks { print "Sanity checks...\n"; local($key, $value); local($w); local(%message); # Record messages on a per-unit basis local(%said); # Avoid duplicate messages # Warn about symbols ever used in conditional dependency with no default while (($key, $value) = each(%condsym)) { unless ($hasdefault{$key}) { $w = (split(' ', $shmaster{"\$$key"}))[0]; $message{$w} .= "#$key "; } } # Warn about any undeclared variables. They are all listed in %shunknown, # being the values while the unit where they appear is the key. If the # symbol is defined by any of the special units included or made visible, # then no warning is issued. local($defined); # True if symbol is defined in one unit local($where); # List of units where symbol is defined local($myself); # The name of the current unit if itself special local($visible); # Symbol made visible via a ?V: line foreach $unit (sort keys %shunknown) { foreach $sym (split(' ', $shunknown{$unit})) { $defined = 0; $where = $shmaster{"\$$sym"}; $defined = 1 if $tempmaster{"\$$sym"} =~ /$unit\b/; $myself = substr($unit, 0, 1) =~ /^[A-Z]/ ? $unit : ''; # Symbol has to be either defined within one of the special units # listed in the dependencies or exported via a ?V: line. unless ($defined) { $defined = &visible($sym, $unit); $spneeded{$unit}++ if $defined; } $message{$unit} .= "\$$sym " unless $defined; } } # Warn about any undeclared files. Files used in one unit are all within # the %fileused table, indexed by unit. If a file is used, it must either # be in the unit that declared it (relying on %filemaster for that) or # the unit listed in %filemaster must be part of our dependency. %said = (); foreach $unit (sort keys %fileused) { foreach $file (split(' ', $fileused{$unit})) { $defined = 0; $where = $filemaster{$file}; # Where file is created $defined = 1 if $unit eq $where; # We're in the unit defining it # Private UU files may be only be created by special units foreach $special (split(' ', $shspecial{$unit})) { last if $defined; $defined = 1 if $where eq $special; } # Exceptions to above rule possible via a ?LINT:create hint, # so parse all known dependencies for the unit... foreach $depend (split(' ', $shdepend{$unit})) { last if $defined; $defined = 1 if $where eq $depend; } $message{$unit} .= "\@$file " unless $defined || $said{"$unit/$file"}++; # Unknown file } } undef %fileused; # Warn about any misused files, kept in %filemisused foreach $unit (sort keys %filemisused) { foreach $file (split(' ', $filemisused{$unit})) { next unless defined $filemaster{$file}; # Skip non UU-files $message{$unit} .= "\@\@$file "; # Misused file } } undef %filemisused; # Warn about temporary files which could be created and inadvertently # override a private UU file (listed in %filemaster). foreach $tmpfile (keys %filesetin) { next unless defined $filemaster{$tmpfile}; $where = $filemaster{$tmpfile}; foreach $unit (split(' ', $filesetin{$tmpfile})) { $message{$unit} .= "\@\@\@$where:$tmpfile "; } } undef %filesetin; # Warn about any set variable which was not listed. foreach $unit (sort keys %shset) { symbol: foreach $sym (split(' ', $shset{$unit})) { next if $shvisible{$sym}; $defined = 0; # Symbol has to be either defined within one of the special units # listed in the dependencies or exported read-write via a ?V: line. # If symbol is exported read-only, report the attempt to set it. $where = $shmaster{"\$$sym"}; study $where; foreach $special (split(' ', $shspecial{$unit})) { $defined = 1 if $where =~ /\b$special\b/; last if $defined; } $visible = 0; $defined = $visible = &visible($sym, $unit) unless $defined; if ($visible && $shvisible{"\$$sym"} ne '') { # We are allowed to set a read-only symbol in the unit which # declared it... next symbol if $shvisible{"\$$sym"} eq $unit; $message{$unit} .= "\&$sym "; # Read-only symbol set next symbol; } $message{$unit} .= "$sym " unless $defined; } } # Warn about any obsolete variable which may be used foreach $unit (sort keys %shused) { foreach $sym (split(' ', $shused{$unit})) { $message{$unit} .= "!$sym " if $Obsolete{$sym} ne ''; } } # Warn about stale dependencies, and prepare successor and predecessor # tables for later topological sort. local($targets, $deps); local(%Succ); # Successors local(%Prec); # Predecessors # Split dependencies and build successors array. foreach $make (@make) { ($targets, $deps) = $make =~ m|(.*):\s*(.*)|; $deps =~ s/\+(\w)/$1/g; # Remove conditional targets foreach $target (split(' ', $targets)) { $Succ{$target} .= $deps . ' '; } } # Special setup for the End target, which normally has a $W dependency for # wanted symbols. In order to detect all the possible cycles, we forge a # huge dependency by making ALL the regular symbols (i.e. those whose first # letter is not uppercased) wanted. local($allwant) = ''; { local($sym, $val); while (($sym, $val) = each %shmaster) { $sym =~ s/^\$//; $allwant .= "$sym " if $val ne ''; } } $Succ{'End'} =~ s/\$W/$allwant/; # Initialize precursors, and spot symbols impossible to 'make', i.e. those # symbols listed in the successors and with no 'make' target. The data # structures %Prec and %Succ will also be used by the cycle lookup code, # in other words, the topological sort. foreach $target (keys %Succ) { $Prec{$target} += 0; # Ensure key is recorded without disturbing. foreach $succ (split(' ', $Succ{$target})) { $Prec{$succ}++; # Successor has one more precursor unless (defined $Succ{$succ} || $said{$succ}++) { foreach $unit (split(' ', $symdep{$succ})) { $message{$unit} .= "?$succ "; # Stale ?MAKE: dependency } } } } undef %symdep; # Check all ?M: dependencies to spot stale ones %said = (); while (($key, $value) = each(%msym)) { next if $value eq ''; # Value is unit name where ?M: occurred foreach $sym (split(' ', $mdep{$key})) { # Loop on C dependencies next if $cmaster{$sym} || $said{$sym}; $message{$value} .= "??$sym "; # Stale ?M: dependency $said{$sym}++; } } undef %said; undef %mdep; undef %msym; # Now actually emit all the warnings local($uv); # Unit defining visible symbol or private file local($w); # Were we are signaling an error foreach $unit (sort keys %message) { undef %said; $w = "\"$unit.U\""; foreach (split(' ', $message{$unit})) { if (s/^#//) { warn "$w: symbol '\$$_' has no default value.\n"; } elsif (s/^\?\?//) { warn "$w: stale ?M: dependency '$_'.\n"; } elsif (s/^\?//) { warn "$w: stale ?MAKE: dependency '$_'.\n"; } elsif (s/^\$//) { if ($shmaster{"\$$_"} ne '') { warn "$w: symbol '\$$_' missing from ?MAKE.\n"; } elsif (($uv = $shvisible{$_}) ne '') { warn "$w: missing $uv from ?MAKE for visible '\$$_'.\n"; } elsif (($uv = $shvisible{"\$$_"}) ne '') { warn "$w: missing $uv from ?MAKE for visible '\$$_'.\n"; } else { warn "\"$unit.U\": unknown symbol '\$$_'.\n"; } ++$said{$_}; } elsif (s/^\&//) { warn "\"$unit.U\": read-only symbol '\$$_' is set.\n"; ++$said{$_}; } elsif (s/^!//) { warn "\"$unit.U\": obsolete symbol '$_' is used.\n"; ++$said{$_}; } elsif (s/^\@\@\@//) { $uv = '?'; # To spot format errors s/^(\w+):// && ($uv = $1); warn "$w: local file '$_' may override the one set by $uv.U.\n"; } elsif (s/^\@\@//) { $uv = $filemaster{$_}; warn "$w: you might not always get file '$_' from $uv.U.\n"; } elsif (s/^\@//) { if ($uv = $filemaster{$_}) { warn "$w: missing $uv from ?MAKE for private file '$_'.\n"; } else { warn "$w: unknown private file '$_'.\n"; } ++$said{"\@$_"}; } else { warn "\"$unit.U\": undeclared symbol '\$$_' is set.\n" unless $said{$_}; } } } # Memory cleanup undef %message; undef %said; undef %shused; undef %shset; undef %shspecial; undef %shvisible; undef %filemaster; # Spot multiply defined C symbols foreach $sym (keys %cmaster) { @sym = split(' ', $cmaster{$sym}); if (@sym > 1) { warn "C symbol '$sym' is defined in the following units:\n"; foreach (@sym) { print STDERR "\t$_.U\n"; } } } undef %cmaster; # Memory cleanup # Warn about multiply defined symbols. There are three kind of symbols: # target symbols, obsolete symbols and temporary symbols. # For each of these sets, we make sure the intersection with the other sets # is empty. Besides, we make sure target symbols are only defined once. local(@sym); foreach $sym (keys %shmaster) { @sym = split(' ', $shmaster{$sym}); if (@sym > 1) { warn "Shell symbol '$sym' is defined in the following units:\n"; foreach (@sym) { print STDERR "\t$_.U\n"; } } $message{$sym} .= 'so ' if $Obsolete{$sym}; $message{$sym} .= 'st ' if $tempmaster{$sym}; } foreach $sym (keys %tempmaster) { $message{$sym} .= 'ot ' if $Obsolete{$sym}; } local($_); while (($sym, $_) = each %message) { if (/so/) { if (/ot/) { warn "Shell symbol '$sym' is altogether:\n"; @sym = split(' ', $shmaster{$sym}); @sym = grep(s/$/.U/, @sym); print STDERR "...defined in: ", join(', ', @sym), "\n"; print STDERR "...obsoleted by $Obsolete{$sym}.\n"; @sym = split(' ', $tempmaster{$sym}); @sym = grep(s/$/.U/, @sym); print STDERR "...used as temporary in:", join(', ', @sym), "\n"; } else { warn "Shell symbol '$sym' is both defined and obsoleted:\n"; @sym = split(' ', $shmaster{$sym}); @sym = grep(s/$/.U/, @sym); print STDERR "...defined in: ", join(', ', @sym), "\n"; print STDERR "...obsoleted by $Obsolete{$sym}.\n"; } } elsif (/st/) { # Cannot be ot as it would imply so warn "Shell symbol '$sym' is both defined and used as temporary:\n"; @sym = split(' ', $shmaster{$sym}); @sym = grep(s/$/.U/, @sym); print STDERR "...defined in: ", join(', ', @sym), "\n"; @sym = split(' ', $tempmaster{$sym}); @sym = grep(s/$/.U/, @sym); print STDERR "...used as temporary in:", join(', ', @sym), "\n"; } elsif (/ot/) { warn "Shell symbol '$sym' obsoleted also used as temporary:\n"; print STDERR "...obsoleted by $Obsolete{$sym}.\n"; @sym = split(' ', $tempmaster{$sym}); @sym = grep(s/$/.U/, @sym); print STDERR "...used as temporary in:", join(', ', @sym), "\n"; } } # Spot multiply defined files, either private or public ones foreach $file (keys %prodfile) { @sym = split(' ', $prodfile{$file}); if (@sym > 1) { warn "File '$file' is defined in the following units:\n"; foreach (@sym) { print STDERR "\t$_\n"; } } } undef %prodfile; # Memory cleanup (we still need %shmaster for tsort) undef %message; undef %tempmaster; undef %Obsolete; # Make sure there is no dependency cycle print "Looking for dependency cycles...\n"; &tsort(*Succ, *Prec); # Destroys info from %Prec } # Make sure last declaration ended correctly with a ?S:. or ?C:. line. # The variable '$where' was correctly positionned by the calling routine. sub check_last_declaration { warn "$where: definition of '\$$s_symbol' not closed by '?S:.'.\n" if $s_symbol ne ''; warn "$where: definition of '$c_symbol' not closed by '?C:.'.\n" if $c_symbol ne ''; warn "$where: magic definition of '$m_symbol' not closed by '?M:.'.\n" if $m_symbol ne ''; $s_symbol = $c_symbol = $m_symbol = ''; } # Make sure the variable is mentionned on the ?MAKE line, if possible in the # definition section. # The variable '$where' was correctly positionned by the calling routine. sub check_definition { local($var) = @_; warn "$where: variable '\$$var' not even listed on ?MAKE: line.\n" unless $defseen{$var} || $condseen{$var} || $depseen{$var}; warn "$where: variable '\$$var' is defined externally.\n" if !$lintextern{$var} && !$defseen{$var} && &wanted($var); } # Is symbol declared somewhere? sub declared { &defined($_[0]) || &wanted($_[0]); } # Is symbol defined by unit? sub defined { $tempseen{$_[0]} || $defseen{$_[0]} || $lintseen{$_[0]}; } # Is symbol wanted by unit? sub wanted { $depseen{$_[0]} || $condseen{$_[0]}; } # Is symbol visible from the unit? # Locate visible symbols throughout the special units. Each unit having # some special dependencies (special units wanted) have an entry in the # %shspecial array, listing all those special dependencies. And each # symbol made visible by ONE special unit has an entry in the %shvisible # array. sub visible { local($symbol, $unit) = @_; local(%explored); # Special units we've already explored &explore($symbol, $unit); # Perform recursive search } # Recursively explore the dependencies to locate a visible symbol sub explore { local($symbol, $unit) = @_; # If unit was already explored, we know it has not been found by following # that path. return 0 if defined $explored{$unit}; $explored{$unit} = 0; # Assume nothing found in this unit local($specials) = $shspecial{$unit}; # Don't waste any time if unit does not have any special units listed # in its dependencies. return 0 unless $specials; foreach $special (split(' ', $specials)) { return 1 if ( $shvisible{"\$$symbol"} eq $unit || $shvisible{$symbol} eq $unit || &explore($symbol, $special) ); } 0; }