| #! /usr/bin/perl -w |
| # |
| # Copyright 2000 Patrik Stridvall |
| # |
| # This library is free software; you can redistribute it and/or |
| # modify it under the terms of the GNU Lesser General Public |
| # License as published by the Free Software Foundation; either |
| # version 2.1 of the License, or (at your option) any later version. |
| # |
| # This library 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 |
| # Lesser General Public License for more details. |
| # |
| # You should have received a copy of the GNU Lesser General Public |
| # License along with this library; if not, write to the Free Software |
| # Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| # |
| |
| use strict; |
| |
| my $name0=$0; |
| $name0 =~ s%^.*/%%; |
| |
| my $invert = 0; |
| my $pattern; |
| my @files = (); |
| my $usage; |
| |
| while(defined($_ = shift)) { |
| if (/^-v$/) { |
| $invert = 1; |
| } elsif (/^--?(\?|h|help)$/) { |
| $usage=0; |
| } elsif (/^-/) { |
| print STDERR "$name0:error: unknown option '$_'\n"; |
| $usage=2; |
| last; |
| } elsif(!defined($pattern)) { |
| $pattern = $_; |
| } else { |
| push @files, $_; |
| } |
| } |
| if (defined $usage) |
| { |
| print "Usage: $name0 [--help] [-v] pattern files...\n"; |
| print "where:\n"; |
| print "--help Prints this help message\n"; |
| print "-v Return functions that do not match pattern\n"; |
| print "pattern A regular expression for the function name\n"; |
| print "files... A list of files to search the function in\n"; |
| exit $usage; |
| } |
| |
| foreach my $file (@files) { |
| open(IN, "< $file") || die "Error: Can't open $file: $!\n"; |
| |
| my $level = 0; |
| my $extern_c = 0; |
| |
| my $again = 0; |
| my $lookahead = 0; |
| while($again || defined(my $line = <IN>)) { |
| if(!$again) { |
| chomp $line; |
| if($lookahead) { |
| $lookahead = 0; |
| $_ .= "\n" . $line; |
| } else { |
| $_ = $line; |
| } |
| } else { |
| $again = 0; |
| } |
| |
| # remove C comments |
| if(s/^(|.*?[^\/])(\/\*.*?\*\/)(.*)$/$1 $3/s) { |
| $again = 1; |
| next; |
| } elsif(/^(.*?)\/\*/s) { |
| $lookahead = 1; |
| next; |
| } |
| |
| # remove C++ comments |
| while(s/^(.*?)\/\/.*?$/$1\n/s) { $again = 1; } |
| if($again) { next; } |
| |
| # remove empty rows |
| if(/^\s*$/) { next; } |
| |
| # remove preprocessor directives |
| if(s/^\s*\#/\#/m) { |
| if(/^\#[.\n\r]*?\\$/m) { |
| $lookahead = 1; |
| next; |
| } elsif(s/^\#\s*(.*?)(\s+(.*?))?\s*$//m) { |
| next; |
| } |
| } |
| |
| # Remove extern "C" |
| if(s/^\s*extern[\s\n]+"C"[\s\n]+\{//m) { |
| $extern_c = 1; |
| $again = 1; |
| next; |
| } elsif(m/^\s*extern[\s\n]+"C"/m) { |
| $lookahead = 1; |
| next; |
| } |
| |
| if($level > 0) |
| { |
| my $line = ""; |
| while(/^[^\{\}]/) { |
| s/^([^\{\}\'\"]*)//s; |
| $line .= $1; |
| if(s/^\'//) { |
| $line .= "\'"; |
| while(/^./ && !s/^\'//) { |
| s/^([^\'\\]*)//s; |
| $line .= $1; |
| if(s/^\\//) { |
| $line .= "\\"; |
| if(s/^(.)//s) { |
| $line .= $1; |
| if($1 eq "0") { |
| s/^(\d{0,3})//s; |
| $line .= $1; |
| } |
| } |
| } |
| } |
| $line .= "\'"; |
| } elsif(s/^\"//) { |
| $line .= "\""; |
| while(/^./ && !s/^\"//) { |
| s/^([^\"\\]*)//s; |
| $line .= $1; |
| if(s/^\\//) { |
| $line .= "\\"; |
| if(s/^(.)//s) { |
| $line .= $1; |
| if($1 eq "0") { |
| s/^(\d{0,3})//s; |
| $line .= $1; |
| } |
| } |
| } |
| } |
| $line .= "\""; |
| } |
| } |
| |
| if(s/^\{//) { |
| $_ = $'; $again = 1; |
| $line .= "{"; |
| $level++; |
| } elsif(s/^\}//) { |
| $_ = $'; $again = 1; |
| $line .= "}" if $level > 1; |
| $level--; |
| if($level == -1 && $extern_c) { |
| $extern_c = 0; |
| $level = 0; |
| } |
| } |
| |
| next; |
| } elsif(/^class[^\}]*{/) { |
| $_ = $'; $again = 1; |
| $level++; |
| next; |
| } elsif(/^class[^\}]*$/) { |
| $lookahead = 1; |
| next; |
| } elsif(/^typedef[^\}]*;/) { |
| next; |
| } elsif(/(extern\s+|static\s+)? |
| (?:__inline__\s+|__inline\s+|inline\s+)? |
| ((struct\s+|union\s+|enum\s+)?(?:\w+(?:\:\:(?:\s*operator\s*[^\)\s]+)?)?)+((\s*(?:\*|\&))+\s*|\s+)) |
| ((__cdecl|__stdcall|CDECL|VFWAPIV|VFWAPI|WINAPIV|WINAPI|CALLBACK)\s+)? |
| ((?:\w+(?:\:\:)?)+(\(\w+\))?)\s*\(([^\)]*)\)\s* |
| (?:\w+(?:\s*\([^\)]*\))?\s*)*\s* |
| (\{|\;)/sx) |
| { |
| $_ = $'; $again = 1; |
| if($11 eq "{") { |
| $level++; |
| } |
| |
| my $linkage = $1; |
| my $return_type = $2; |
| my $calling_convention = $7; |
| my $name = $8; |
| my $arguments = $10; |
| |
| if(!defined($linkage)) { |
| $linkage = ""; |
| } |
| |
| if(!defined($calling_convention)) { |
| $calling_convention = ""; |
| } |
| |
| $linkage =~ s/\s*$//; |
| |
| $return_type =~ s/\s*$//; |
| $return_type =~ s/\s*\*\s*/*/g; |
| $return_type =~ s/(\*+)/ $1/g; |
| |
| $arguments =~ y/\t\n/ /; |
| $arguments =~ s/^\s*(.*?)\s*$/$1/; |
| if($arguments eq "") { $arguments = "void" } |
| |
| my @argument_types; |
| my @argument_names; |
| my @arguments = split(/,/, $arguments); |
| foreach my $n (0..$#arguments) { |
| my $argument_type = ""; |
| my $argument_name = ""; |
| my $argument = $arguments[$n]; |
| $argument =~ s/^\s*(.*?)\s*$/$1/; |
| # print " " . ($n + 1) . ": '$argument'\n"; |
| $argument =~ s/^(IN OUT(?=\s)|IN(?=\s)|OUT(?=\s)|\s*)\s*//; |
| $argument =~ s/^(const(?=\s)|CONST(?=\s)|__const(?=\s)|__restrict(?=\s)|\s*)\s*//; |
| if($argument =~ /^\.\.\.$/) { |
| $argument_type = "..."; |
| $argument_name = "..."; |
| } elsif($argument =~ /^ |
| ((?:struct\s+|union\s+|enum\s+|(?:signed\s+|unsigned\s+) |
| (?:short\s+(?=int)|long\s+(?=int))?)?(?:\w+(?:\:\:)?)+)\s* |
| ((?:const(?=\s)|CONST(?=\s)|__const(?=\s)|__restrict(?=\s))?\s*(?:\*\s*?)*)\s* |
| (?:const(?=\s)|CONST(?=\s)|__const(?=\s)|__restrict(?=\s))?\s* |
| (\w*)\s* |
| (?:\[\]|\s+OPTIONAL)?/x) |
| { |
| $argument_type = "$1"; |
| if($2 ne "") { |
| $argument_type .= " $2"; |
| } |
| $argument_name = $3; |
| |
| $argument_type =~ s/\s*const\s*/ /; |
| $argument_type =~ s/^\s*(.*?)\s*$/$1/; |
| |
| $argument_name =~ s/^\s*(.*?)\s*$/$1/; |
| } else { |
| die "$file: $.: syntax error: '$argument'\n"; |
| } |
| $argument_types[$n] = $argument_type; |
| $argument_names[$n] = $argument_name; |
| # print " " . ($n + 1) . ": '$argument_type': '$argument_name'\n"; |
| } |
| if($#argument_types == 0 && $argument_types[0] =~ /^void$/i) { |
| $#argument_types = -1; |
| $#argument_names = -1; |
| } |
| |
| @arguments = (); |
| foreach my $n (0..$#argument_types) { |
| if($argument_names[$n] && $argument_names[$n] ne "...") { |
| if($argument_types[$n] !~ /\*$/) { |
| $arguments[$n] = $argument_types[$n] . " " . $argument_names[$n]; |
| } else { |
| $arguments[$n] = $argument_types[$n] . $argument_names[$n]; |
| } |
| } else { |
| $arguments[$n] = $argument_types[$n]; |
| } |
| } |
| |
| $arguments = join(", ", @arguments); |
| if(!$arguments) { $arguments = "void"; } |
| |
| if((!$invert && $name =~ /$pattern/) || ($invert && $name !~ /$pattern/)) { |
| if($calling_convention) { |
| print "$return_type $calling_convention $name($arguments)\n"; |
| } else { |
| if($return_type =~ /\*$/) { |
| print "$return_type$name($arguments)\n"; |
| } else { |
| print "$return_type $name($arguments)\n"; |
| } |
| } |
| } |
| } elsif(/\'(?:[^\\\']*|\\.)*\'/s) { |
| $_ = $'; $again = 1; |
| } elsif(/\"(?:[^\\\"]*|\\.)*\"/s) { |
| $_ = $'; $again = 1; |
| } elsif(/;/s) { |
| $_ = $'; $again = 1; |
| } elsif(/extern\s+"C"\s+{/s) { |
| $_ = $'; $again = 1; |
| } elsif(/\{/s) { |
| $_ = $'; $again = 1; |
| $level++; |
| } else { |
| $lookahead = 1; |
| } |
| } |
| close(IN); |
| } |