#!/usr/bin/perl | |
# | |
# Program: find-cycles.pl | |
# | |
# Synopsis: Given a list of possibly cyclic dependencies, merge all the | |
# cycles. This makes it possible to topologically sort the | |
# dependencies between different parts of LLVM. | |
# | |
# Syntax: find-cycles.pl < LibDeps.txt > FinalLibDeps.txt | |
# | |
# Input: cycmem1: cycmem2 dep1 dep2 | |
# cycmem2: cycmem1 dep3 dep4 | |
# boring: dep4 | |
# | |
# Output: cycmem1 cycmem2: dep1 dep2 dep3 dep4 | |
# boring: dep4 | |
# | |
# This file was written by Eric Kidd, and is placed into the public domain. | |
# | |
use 5.006; | |
use strict; | |
use warnings; | |
my %DEPS; | |
my @CYCLES; | |
sub find_all_cycles; | |
# Read our dependency information. | |
while (<>) { | |
chomp; | |
my ($module, $dependency_str) = /^\s*([^:]+):\s*(.*)\s*$/; | |
die "Malformed data: $_" unless defined $dependency_str; | |
my @dependencies = split(/ /, $dependency_str); | |
$DEPS{$module} = \@dependencies; | |
} | |
# Partition our raw dependencies into sets of cyclically-connected nodes. | |
find_all_cycles(); | |
# Print out the finished cycles, with their dependencies. | |
my @output; | |
my $cycles_found = 0; | |
foreach my $cycle (@CYCLES) { | |
my @modules = sort keys %{$cycle}; | |
# Merge the dependencies of all modules in this cycle. | |
my %dependencies; | |
foreach my $module (@modules) { | |
@dependencies{@{$DEPS{$module}}} = 1; | |
} | |
# Prune the known cyclic dependencies. | |
foreach my $module (@modules) { | |
delete $dependencies{$module}; | |
} | |
# Warn about possible linker problems. | |
my @archives = grep(/\.a$/, @modules); | |
if (@archives > 1) { | |
$cycles_found = $cycles_found + 1; | |
print STDERR "find-cycles.pl: Circular dependency between *.a files:\n"; | |
print STDERR "find-cycles.pl: ", join(' ', @archives), "\n"; | |
push @modules, @archives; # WORKAROUND: Duplicate *.a files. Ick. | |
} elsif (@modules > 1) { | |
$cycles_found = $cycles_found + 1; | |
print STDERR "find-cycles.pl: Circular dependency between *.o files:\n"; | |
print STDERR "find-cycles.pl: ", join(' ', @modules), "\n"; | |
push @modules, @modules; # WORKAROUND: Duplicate *.o files. Ick. | |
} | |
# Add to our output. (@modules is already as sorted as we need it to be.) | |
push @output, (join(' ', @modules) . ': ' . | |
join(' ', sort keys %dependencies) . "\n"); | |
} | |
print sort @output; | |
exit $cycles_found; | |
#========================================================================== | |
# Depedency Cycle Support | |
#========================================================================== | |
# For now, we have cycles in our dependency graph. Ideally, each cycle | |
# would be collapsed down to a single *.a file, saving us all this work. | |
# | |
# To understand this code, you'll need a working knowledge of Perl 5, | |
# and possibly some quality time with 'man perlref'. | |
my %SEEN; | |
my %CYCLES; | |
sub find_cycles ($@); | |
sub found_cycles ($@); | |
sub find_all_cycles { | |
# Find all multi-item cycles. | |
my @modules = sort keys %DEPS; | |
foreach my $module (@modules) { find_cycles($module); } | |
# Build fake one-item "cycles" for the remaining modules, so we can | |
# treat them uniformly. | |
foreach my $module (@modules) { | |
unless (defined $CYCLES{$module}) { | |
my %cycle = ($module, 1); | |
$CYCLES{$module} = \%cycle; | |
} | |
} | |
# Find all our unique cycles. We have to do this the hard way because | |
# we apparently can't store hash references as hash keys without making | |
# 'strict refs' sad. | |
my %seen; | |
foreach my $cycle (values %CYCLES) { | |
unless ($seen{$cycle}) { | |
$seen{$cycle} = 1; | |
push @CYCLES, $cycle; | |
} | |
} | |
} | |
# Walk through our graph depth-first (keeping a trail in @path), and report | |
# any cycles we find. | |
sub find_cycles ($@) { | |
my ($module, @path) = @_; | |
if (str_in_list($module, @path)) { | |
found_cycle($module, @path); | |
} else { | |
return if defined $SEEN{$module}; | |
$SEEN{$module} = 1; | |
foreach my $dep (@{$DEPS{$module}}) { | |
find_cycles($dep, @path, $module); | |
} | |
} | |
} | |
# Give a cycle, attempt to merge it with pre-existing cycle data. | |
sub found_cycle ($@) { | |
my ($module, @path) = @_; | |
# Pop any modules which aren't part of our cycle. | |
while ($path[0] ne $module) { shift @path; } | |
#print join("->", @path, $module) . "\n"; | |
# Collect the modules in our cycle into a hash. | |
my %cycle; | |
foreach my $item (@path) { | |
$cycle{$item} = 1; | |
if (defined $CYCLES{$item}) { | |
# Looks like we intersect with an existing cycle, so merge | |
# all those in, too. | |
foreach my $old_item (keys %{$CYCLES{$item}}) { | |
$cycle{$old_item} = 1; | |
} | |
} | |
} | |
# Update our global cycle table. | |
my $cycle_ref = \%cycle; | |
foreach my $item (keys %cycle) { | |
$CYCLES{$item} = $cycle_ref; | |
} | |
#print join(":", sort keys %cycle) . "\n"; | |
} | |
sub str_in_list ($@) { | |
my ($str, @list) = @_; | |
foreach my $item (@list) { | |
return 1 if ($item eq $str); | |
} | |
return 0; | |
} |