aboutsummaryrefslogtreecommitdiffstats
path: root/strip_and_check_elf_files
blob: 4cebbe9e42799ddb3358070eada6f75878cf3039 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
#!/usr/bin/perl
# $Id$
# Strip files

use strict;
use warnings;
use File::Find;

my $buildroot = $ENV{RPM_BUILD_ROOT};
die "No build root defined" unless $buildroot;
die "Invalid build root" unless -d $buildroot;
# normalize build root
$buildroot =~ s|/$||;

# set LIBRARY_PATH to check libraries in build root
my $lib = `rpm --eval %_lib`;
chomp $lib;
$ENV{LD_LIBRARY_PATH}="$buildroot/$lib:$buildroot/usr/$lib";

my (@shared_libs, @executables, @static_libs);
find(\&keep_wanted, $buildroot);

strip_files() if !$ENV{DONT_STRIP};
check_missing_or_unused_libs();

sub strip_files {
    my @to_strip = (@shared_libs, @executables);

    if ($ENV{EXCLUDE_FROM_STRIP}) {
        my $exclude_pattern = join('|', split(/\s+/, $ENV{EXCLUDE_FROM_STRIP}));
        my $compiled_pattern = qr/($exclude_pattern)/;
        @to_strip = grep { !/$compiled_pattern/ } @to_strip;
    }

    system(
        "strip",
        "--remove-section=.comment",
        "--remove-section=.note",
        $_) foreach @to_strip;
}

sub check_missing_or_unused_libs {
    foreach my $f (@shared_libs, @executables) {
        my (undef, undef, @l) = `ldd -u -r $f 2>/dev/null`;
        @l or next;
        my $f_ = substr($f, length($buildroot));
        print STDERR
            "Warning: unused libraries in $f_: ",
            join(' ', map { basename($_) } @l), "\n";
    }
    foreach my $f (@shared_libs) {
        my @l = `ldd -r $f 2>&1 >/dev/null` or next;
        my $f_ = substr($f, length($buildroot));
        print STDERR
            "Warning: undefined symbols in $f_: ",
            join(' ', map { /undefined symbol: (\S+)/ ? $1 : () } @l), "\n";
    }
}

# TODO: we should write a binding for libfile...
sub expensive_test {
    my ($file) = @_;
    my $type = `file -- $file`;
}

# Check if a file is an elf binary, shared library, or static library,
# for use by File::Find. It'll fill the following 3 arrays with anything
# it finds:
sub keep_wanted() {
    # skip everything but files
    return unless -f $_;
    # skip symlinks
    return if -l $_;
    return if $File::Find::dir =~ m!/usr/lib/debug($|/)!;

    # Does its filename look like a shared library?
    if (m/\.so/) {
        # Ok, do the expensive test.
        if (expensive_test($_) =~ m/ELF.*shared/) {
            push @shared_libs, $File::Find::name;
            return;
        }
    }
    
    # Is it executable? -x isn't good enough, so we need to use stat.
    my (undef, undef, $mode, undef) = stat(_);
    if ($mode & 0111) {
        # Ok, expensive test.
        if (expensive_test($_) =~ m/ELF.*executable/) {
            push @executables, $File::Find::name;
            return;
        }
    }
    
    # Is it a static library, and not a debug library?
    if (m/lib.*\.a/ && ! m/_g\.a/) {
        push @static_libs, $File::Find::name;
        return;
    }
}

sub basename {
    local $_ = shift;
    s|/*\s*$||; s|.*/||;
    $_
}