1#!/usr/bin/perl -w 2 3# Copyright (C) 2005, 2006 Apple Computer, Inc. All rights reserved. 4# 5# Redistribution and use in source and binary forms, with or without 6# modification, are permitted provided that the following conditions 7# are met: 8# 9# 1. Redistributions of source code must retain the above copyright 10# notice, this list of conditions and the following disclaimer. 11# 2. Redistributions in binary form must reproduce the above copyright 12# notice, this list of conditions and the following disclaimer in the 13# documentation and/or other materials provided with the distribution. 14# 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of 15# its contributors may be used to endorse or promote products derived 16# from this software without specific prior written permission. 17# 18# THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY 19# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 20# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 21# DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY 22# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 23# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 24# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND 25# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 26# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 27# THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 28 29# "report-include-statistics" script for WebKit Open Source Project 30 31use strict; 32use File::Find; 33 34find(\&wanted, @ARGV ? @ARGV : "."); 35 36my %paths; 37my %sources; 38my %includes; 39 40sub wanted 41{ 42 my $file = $_; 43 44 if ($file eq "icu") { 45 $File::Find::prune = 1; 46 return; 47 } 48 49 if ($file !~ /^\./ && $file =~ /\.(h|cpp|c|mm|m)$/) { 50 $paths{$file} = $File::Find::name; 51 $sources{$file} = $File::Find::name if $file !~ /\.h/; 52 open FILE, $file or die; 53 while (<FILE>) { 54 if (m-^\s*#\s*(include|import)\s+["<]((\S+/)*)(\S+)[">]-) { 55 my $include = ($2 eq "sys/" ? $2 : "") . $4; 56 $includes{$file}{$include}++; 57 } 58 } 59 close FILE; 60 } 61} 62 63my %totalIncludes; 64 65sub fillOut 66{ 67 my ($file) = @_; 68 69 return if defined $totalIncludes{$file}; 70 71 for my $include (keys %{ $includes{$file} }) { 72 $totalIncludes{$file}{$include} = 1; 73 fillOut($include); 74 for my $i (keys %{ $totalIncludes{$include} }) { 75 $totalIncludes{$file}{$i} = 1; 76 } 77 } 78} 79 80my %inclusionCounts; 81for my $file (keys %includes) { 82 $inclusionCounts{$file} = 0; 83 fillOut($file); 84} 85 86for my $file (keys %sources) { 87 for my $include (keys %{ $totalIncludes{$file} }) { 88 $inclusionCounts{$include}++; 89 } 90} 91 92for my $file (sort mostincludedcmp keys %includes) { 93 next if !$paths{$file}; 94 my $count = $inclusionCounts{$file}; 95 my $numIncludes = keys %{ $includes{$file} }; 96 my $numTotalIncludes = keys %{ $totalIncludes{$file} }; 97 print "$file is included $count times, includes $numIncludes files directly, $numTotalIncludes files total.\n" 98} 99 100# Sort most-included files first. 101sub mostincludedcmp($$) 102{ 103 my ($filea, $fileb) = @_; 104 105 my $counta = $inclusionCounts{$filea} || 0; 106 my $countb = $inclusionCounts{$fileb} || 0; 107 return $countb <=> $counta if $counta != $countb; 108 109 my $ta = keys %{ $totalIncludes{$filea} }; 110 my $tb = keys %{ $totalIncludes{$fileb} }; 111 return $ta <=> $tb if $ta != $tb; 112 113 return $filea cmp $fileb; 114} 115