blob: be7af22df43656c956bbe6d958b78ce0ae596837 [file] [log] [blame]
Reid Spencer271ed742004-12-30 23:07:56 +00001#!/usr/bin/perl -w
2#
3# Program: GenLibDeps.pl
4#
5# Synopsis: Generate HTML output that shows the dependencies between a set of
6# libraries. The output of this script should periodically replace
7# the similar content in the UsingLibraries.html document.
8#
Reid Spencereca6f732006-07-27 23:00:30 +00009# Syntax: GenLibDeps.pl [-flat] <directory_with_libraries_in_it> [path_to_nm_binary]
Reid Spencer271ed742004-12-30 23:07:56 +000010#
11
Reid Spencer89f33a12006-03-19 22:08:01 +000012# Parse arguments...
Reid Spencer238caef2006-07-25 19:12:06 +000013my $FLAT = 0;
14my $WHY = 0;
Reid Spencer89f33a12006-03-19 22:08:01 +000015while (scalar(@ARGV) and ($_ = $ARGV[0], /^[-+]/)) {
16 shift;
17 last if /^--$/; # Stop processing arguments on --
18
19 # List command line options here...
20 if (/^-flat$/) { $FLAT = 1; next; }
Reid Spencer238caef2006-07-25 19:12:06 +000021 if (/^-why/) { $WHY = 1; $FLAT = 1; next; }
Reid Spencer89f33a12006-03-19 22:08:01 +000022 print "Unknown option: $_ : ignoring!\n";
23}
24
Reid Spencer271ed742004-12-30 23:07:56 +000025# Give first option a name.
26my $Directory = $ARGV[0];
Reid Spencereca6f732006-07-27 23:00:30 +000027my $nmPath = $ARGV[1];
Reid Spencer271ed742004-12-30 23:07:56 +000028
Reid Spencere153fb32005-01-05 17:29:29 +000029# Find the "dot" program
Reid Spencer11b00232006-05-13 02:48:45 +000030my $DotPath="";
Reid Spencer5e6f97b2006-04-20 23:09:57 +000031if (!$FLAT) {
Reid Spencer11b00232006-05-13 02:48:45 +000032 chomp($DotPath = `which dot`);
Reid Spencer5e6f97b2006-04-20 23:09:57 +000033 die "Can't find 'dot'" if (! -x "$DotPath");
34}
Reid Spencere153fb32005-01-05 17:29:29 +000035
Reid Spencereca6f732006-07-27 23:00:30 +000036if ($nmPath eq "") {
37 chomp($nmPath=`which nm`);
38 die "Can't find 'nm'" if (! -x "$nmPath");
39}
Reid Spencer11b00232006-05-13 02:48:45 +000040
Reid Spencer271ed742004-12-30 23:07:56 +000041# Open the directory and read its contents, sorting by name and differentiating
42# by whether its a library (.a) or an object file (.o)
43opendir DIR,$Directory;
44my @files = readdir DIR;
45closedir DIR;
46@libs = grep(/libLLVM.*\.a$/,sort(@files));
47@objs = grep(/LLVM.*\.o$/,sort(@files));
48
49# Declare the hashes we will use to keep track of the library and object file
50# symbol definitions.
51my %libdefs;
52my %objdefs;
53
54# Gather definitions from the libraries
55foreach $lib (@libs ) {
56 open DEFS,
Reid Spencer11b00232006-05-13 02:48:45 +000057 "$nmPath -g $Directory/$lib | grep ' [ABCDGRST] ' | sed -e 's/^[0-9A-Fa-f]* [ABCDGRST] //' | sort | uniq |";
Reid Spencer271ed742004-12-30 23:07:56 +000058 while (<DEFS>) {
59 chomp($_);
60 $libdefs{$_} = $lib;
61 }
62 close DEFS;
63}
64
65# Gather definitions from the object files.
66foreach $obj (@objs ) {
67 open DEFS,
Reid Spencer11b00232006-05-13 02:48:45 +000068 "$nmPath -g $Directory/$obj | grep ' [ABCDGRST] ' | sed -e 's/^[0-9A-Fa-f]* [ABCDGRST] //' | sort | uniq |";
Reid Spencer271ed742004-12-30 23:07:56 +000069 while (<DEFS>) {
70 chomp($_);
71 $objdefs{$_} = $obj;
72 }
73 close DEFS;
74}
75
76# Generate one entry in the <dl> list. This generates the <dt> and <dd> elements
77# for one library or object file. The <dt> provides the name of the library or
78# object. The <dd> provides a list of the libraries/objects it depends on.
79sub gen_one_entry {
80 my $lib = $_[0];
Reid Spencere153fb32005-01-05 17:29:29 +000081 my $lib_ns = $lib;
82 $lib_ns =~ s/(.*)\.[oa]/$1/;
Reid Spencer89f33a12006-03-19 22:08:01 +000083 if ($FLAT) {
84 print "$lib:";
Reid Spencer238caef2006-07-25 19:12:06 +000085 if ($WHY) { print "\n"; }
Reid Spencer89f33a12006-03-19 22:08:01 +000086 } else {
87 print " <dt><b>$lib</b</dt><dd><ul>\n";
88 }
Reid Spencer271ed742004-12-30 23:07:56 +000089 open UNDEFS,
Reid Spencer11b00232006-05-13 02:48:45 +000090 "$nmPath -g -u $Directory/$lib | sed -e 's/^ *U //' | sort | uniq |";
Reid Spencer238caef2006-07-25 19:12:06 +000091 my %DepLibs;
Reid Spencer271ed742004-12-30 23:07:56 +000092 while (<UNDEFS>) {
93 chomp;
Reid Spencer238caef2006-07-25 19:12:06 +000094 my $lib_printed = 0;
Reid Spencer271ed742004-12-30 23:07:56 +000095 if (defined($libdefs{$_}) && $libdefs{$_} ne $lib) {
Reid Spencer238caef2006-07-25 19:12:06 +000096 $DepLibs{$libdefs{$_}} = [] unless exists $DepLibs{$libdefs{$_}};
97 push(@{$DepLibs{$libdefs{$_}}}, $_);
Reid Spencer271ed742004-12-30 23:07:56 +000098 } elsif (defined($objdefs{$_}) && $objdefs{$_} ne $lib) {
99 $libroot = $lib;
100 $libroot =~ s/lib(.*).a/$1/;
101 if ($objdefs{$_} ne "$libroot.o") {
Reid Spencer238caef2006-07-25 19:12:06 +0000102 $DepLibs{$objdefs{$_}} = [] unless exists $DepLibs{$objdefs{$_}};
103 push(@{$DepLibs{$objdefs{$_}}}, $_);
Reid Spencer271ed742004-12-30 23:07:56 +0000104 }
105 }
106 }
107 close UNDEFS;
Reid Spencer238caef2006-07-25 19:12:06 +0000108 for my $key (sort keys %DepLibs) {
Reid Spencer89f33a12006-03-19 22:08:01 +0000109 if ($FLAT) {
Reid Spencer238caef2006-07-25 19:12:06 +0000110 print " $key";
111 if ($WHY) {
112 print "\n";
113 my @syms = @{$DepLibs{$key}};
114 foreach $sym (@syms) {
115 print " $sym\n";
116 }
117 }
Reid Spencer89f33a12006-03-19 22:08:01 +0000118 } else {
Reid Spencer238caef2006-07-25 19:12:06 +0000119 print " <li>$key</li>\n";
Reid Spencer89f33a12006-03-19 22:08:01 +0000120 }
Reid Spencer238caef2006-07-25 19:12:06 +0000121 $suffix = substr($key,length($key)-1,1);
122 $key =~ s/(.*)\.[oa]/$1/;
Reid Spencere153fb32005-01-05 17:29:29 +0000123 if ($suffix eq "a") {
Reid Spencer238caef2006-07-25 19:12:06 +0000124 if (!$FLAT) { print DOT "$lib_ns -> $key [ weight=0 ];\n" };
Reid Spencere153fb32005-01-05 17:29:29 +0000125 } else {
Reid Spencer238caef2006-07-25 19:12:06 +0000126 if (!$FLAT) { print DOT "$lib_ns -> $key [ weight=10];\n" };
Reid Spencere153fb32005-01-05 17:29:29 +0000127 }
Reid Spencer271ed742004-12-30 23:07:56 +0000128 }
Reid Spencer89f33a12006-03-19 22:08:01 +0000129 if ($FLAT) {
Reid Spencer238caef2006-07-25 19:12:06 +0000130 if (!$WHY) {
131 print "\n";
132 }
Reid Spencer89f33a12006-03-19 22:08:01 +0000133 } else {
134 print " </ul></dd>\n";
135 }
Reid Spencer271ed742004-12-30 23:07:56 +0000136}
137
138# Make sure we flush on write. This is slower but correct based on the way we
139# write I/O in gen_one_entry.
140$| = 1;
141
142# Print the definition list tag
Reid Spencer89f33a12006-03-19 22:08:01 +0000143if (!$FLAT) {
Reid Spencer5e6f97b2006-04-20 23:09:57 +0000144 print "<dl>\n";
145
146 open DOT, "| $DotPath -Tgif > libdeps.gif";
147
148 print DOT "digraph LibDeps {size=\"40,15\"; ratio=\"1.33333\"; margin=\"0.25\"; rankdir=\"LR\"; mclimit=\"50.0\"; ordering=\"out\"; center=\"1\";\n";
149 print DOT "node [shape=\"box\",color=\"#000088\",fillcolor=\"#FFFACD\",fontcolor=\"#5577DD\",style=\"filled\",fontsize=\"24\"];\n";
150 print DOT "edge [style=\"solid\",color=\"#000088\"];\n";
Reid Spencer89f33a12006-03-19 22:08:01 +0000151}
Reid Spencer271ed742004-12-30 23:07:56 +0000152
153# Print libraries first
154foreach $lib (@libs) {
155 gen_one_entry($lib);
156}
Reid Spencer5e6f97b2006-04-20 23:09:57 +0000157
158if (!$FLAT) {
159 print DOT "}\n";
160 close DOT;
161 open DOT, "| $DotPath -Tgif > objdeps.gif";
162 print DOT "digraph ObjDeps {size=\"40,15\"; ratio=\"1.33333\"; margin=\"0.25\"; rankdir=\"LR\"; mclimit=\"50.0\"; ordering=\"out\"; center=\"1\";\n";
163 print DOT "node [shape=\"box\",color=\"#000088\",fillcolor=\"#FFFACD\",fontcolor=\"#5577DD\",style=\"filled\",fontsize=\"24\"];\n";
164 print DOT "edge [style=\"solid\",color=\"#000088\"];\n";
165}
Reid Spencer271ed742004-12-30 23:07:56 +0000166
167# Print objects second
168foreach $obj (@objs) {
169 gen_one_entry($obj);
170}
171
Reid Spencer5e6f97b2006-04-20 23:09:57 +0000172if (!$FLAT) {
173 print DOT "}\n";
174 close DOT;
Reid Spencere153fb32005-01-05 17:29:29 +0000175
Reid Spencer271ed742004-12-30 23:07:56 +0000176# Print end tag of definition list element
Reid Spencer89f33a12006-03-19 22:08:01 +0000177 print "</dl>\n";
178}