#!/usr/bin/perl # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 2 of the License, or # (at your option) any later version. # # This program 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 General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . use strict; use warnings; use Test::More; use Test::Dpkg qw(:needs :paths); use Config; use Cwd; use IPC::Cmd qw(can_run); if (defined $Config{bin_ELF} && $Config{bin_ELF} eq 'define') { plan tests => 150; } else { plan skip_all => 'only ELF is currently supported'; } $ENV{DEB_BUILD_ARCH} = 'amd64'; $ENV{DEB_HOST_ARCH} = 'amd64'; use_ok('Dpkg::Shlibs'); my $tmp; my @tmp; my %tmp; my $datadir = test_get_data_path(); sub load_objdump_obj { my $name = shift; my $obj = Dpkg::Shlibs::Objdump::Object->new(); open my $objdump, '<', "$datadir/objdump.$name" or die "$datadir/objdump.$name: $!"; $obj->parse_objdump_output($objdump); close $objdump; return $obj; } my @librarypaths; { # XXX: Keep as long as we support the deprecated LD_LIBRARY_PATH. local $ENV{LD_LIBRARY_PATH} = '/test-env'; Dpkg::Shlibs::add_library_dir('/test-a'); @librarypaths = Dpkg::Shlibs::get_library_paths(); is($librarypaths[0], '/test-a', 'add_library_dir() does not get lost'); Dpkg::Shlibs::add_library_dir('/test-b'); @librarypaths = Dpkg::Shlibs::get_library_paths(); is_deeply([ @librarypaths[0, 1] ] , [ '/test-a', '/test-b' ], 'add_library_dir() prepends'); } Dpkg::Shlibs::blank_library_paths(); # We want relative paths inside the ld.so.conf fragments to work, and $srcdir # is usually a relative path, so let's temporarily switch directory. # XXX: An alternative would be to make parse_ldso_conf relative path aware. my $cwd = getcwd(); test_needs_srcdir_switch(); Dpkg::Shlibs::parse_ldso_conf('t/Dpkg_Shlibs/ld.so.conf'); chdir($cwd); @librarypaths = Dpkg::Shlibs::get_library_paths(); is_deeply(\@librarypaths, [ qw( /nonexistent32 /nonexistent/lib64 /usr/local/lib /nonexistent/lib128 ) ], 'parsed library paths'); use_ok('Dpkg::Shlibs::Objdump'); my $obj; $obj = load_objdump_obj('dbd-pg'); ok(!$obj->is_public_library(), 'Pg.so is not a public library'); ok(!$obj->is_executable(), 'Pg.so is not an executable'); $obj = load_objdump_obj('ls'); ok(!$obj->is_public_library(), 'ls is not a public library'); ok($obj->is_executable(), 'ls is an executable'); my $sym = $obj->get_symbol('optarg@GLIBC_2.0'); ok($sym, 'optarg@GLIBC_2.0 exists'); ok(!$sym->{defined}, 'R_*_COPY relocations are taken into account'); # Non-regression test for #506139 $obj = load_objdump_obj('space'); $sym = $obj->get_symbol('singlespace'); ok($sym, 'version less symbol separated by a single space are correctly parsed'); $obj = load_objdump_obj('libc6-2.6'); ok($obj->is_public_library(), 'libc6 is a public library'); ok($obj->is_executable(), 'libc6 is an executable'); is($obj->{SONAME}, 'libc.so.6', 'SONAME'); is($obj->{HASH}, '0x13d99c', 'HASH'); is($obj->{GNU_HASH}, '0x194', 'GNU_HASH'); is($obj->{format}, 'elf32-i386', 'format'); is_deeply($obj->{flags}, { DYNAMIC => 1, HAS_SYMS => 1, D_PAGED => 1 }, 'flags'); is_deeply($obj->{NEEDED}, [ 'ld-linux.so.2' ], 'NEEDED'); is_deeply([ $obj->get_needed_libraries ], [ 'ld-linux.so.2' ], 'NEEDED'); $sym = $obj->get_symbol('_sys_nerr@GLIBC_2.3'); is_deeply($sym, { name => '_sys_nerr', version => 'GLIBC_2.3', soname => 'libc.so.6', objid => 'libc.so.6', section => '.rodata', dynamic => 1, debug => '', type => 'O', weak => '', local => '', global => 1, visibility => '', hidden => 1, defined => 1, }, 'Symbol'); $sym = $obj->get_symbol('_IO_stdin_used'); is_deeply($sym, { name => '_IO_stdin_used', version => '', soname => 'libc.so.6', objid => 'libc.so.6', section => '*UND*', dynamic => 1, debug => '', type => ' ', weak => 1, local => '', global => '', visibility => '', hidden => '', defined => '', }, 'Symbol 2'); my @syms = $obj->get_exported_dynamic_symbols; is(scalar @syms, 2231, 'defined && dynamic'); @syms = $obj->get_undefined_dynamic_symbols; is(scalar @syms, 9, 'undefined && dynamic'); use_ok('Dpkg::Shlibs::SymbolFile'); use_ok('Dpkg::Shlibs::Symbol'); my $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbol_file.tmp"); my $sym_file_dup = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbol_file.tmp"); my $sym_file_old = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbol_file.tmp"); my $obj_old = load_objdump_obj('libc6-2.3'); $sym_file->merge_symbols($obj_old, '2.3.6.ds1-13'); $sym_file_old->merge_symbols($obj_old, '2.3.6.ds1-13'); ok($sym_file->has_object('libc.so.6'), 'SONAME in sym file'); $sym_file->merge_symbols($obj, '2.6-1'); ok($sym_file->get_new_symbols($sym_file_old), 'has new symbols'); ok($sym_file_old->get_lost_symbols($sym_file), 'has lost symbols'); is($sym_file_old->lookup_symbol('__bss_start@Base', ['libc.so.6']), undef, 'internal symbols are ignored'); %tmp = $sym_file->lookup_symbol('_errno@GLIBC_2.0', ['libc.so.6'], 1); isa_ok($tmp{symbol}, 'Dpkg::Shlibs::Symbol'); is_deeply(\%tmp, { symbol => Dpkg::Shlibs::Symbol->new( symbol => '_errno@GLIBC_2.0', minver => '2.3.6.ds1-13', dep_id => 0, deprecated => '2.6-1', ), soname => 'libc.so.6' }, 'deprecated symbol'); # Wildcard test my $pat = $sym_file_old->create_symbol('*@GLIBC_PRIVATE 2.3.6.wildcard'); $sym_file_old->add_symbol($pat, 'libc.so.6'); $sym_file_old->merge_symbols($obj, '2.6-1'); $sym = $sym_file_old->lookup_symbol('__nss_services_lookup@GLIBC_PRIVATE', 'libc.so.6'); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => '__nss_services_lookup@GLIBC_PRIVATE', minver => '2.3.6.wildcard', dep_id => 0, deprecated => 0, tags => { symver => undef, optional => undef, }, tagorder => [ 'symver', 'optional' ], matching_pattern => $pat, ), 'wildcarded symbol'); # Save -> Load test use File::Temp; use File::Basename qw(basename); sub save_load_test { my ($symfile, $comment, @opts) = @_; my $save_file = File::Temp->new(); $symfile->save($save_file->filename, @opts); my $dup = Dpkg::Shlibs::SymbolFile->new(file => $save_file->filename); # Force sync of non-stored attributes $dup->{file} = $symfile->{file}; $dup->{arch} = $symfile->{arch}; is_deeply($dup, $symfile, $comment); if (-f $symfile->{file}) { is(system('diff', '-u', $symfile->{file}, $save_file->filename), 0, basename($symfile->{file}) . ' dumped identical'); } } save_load_test($sym_file, 'save -> load'); # Test ignoring internal symbols $obj = load_objdump_obj('internal'); # Do not ignore any internal symbols $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.internal-filter"); $sym_file->merge_symbols($obj, '100.MISSING'); $sym = $sym_file->lookup_symbol('symbol@Base', ['libinternal.so.0']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol@Base', minver => '1.0', dep_id => 0, deprecated => 0, ), 'symbol unaffected w/o including internal symbols'); $sym = $sym_file->lookup_symbol('.gomp_critical_user_foo@Base', ['libinternal.so.0']); is($sym, undef, 'gomp symbol omitted while filtering internal symbols'); $sym = $sym_file->lookup_symbol('__aeabi_lcmp@GCC_3.0', ['libinternal.so.0']); is($sym, undef, 'known aeabi symbol omitted while filtering internal symbols'); $sym = $sym_file->lookup_symbol('__aeabi_unknown@GCC_4.0', ['libinternal.so.0']); is($sym, undef, 'unknown aeabi symbol omitted while filtering internal symbols'); # Include internal symbols using the allow-internal tag. $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.internal-allow"); $sym_file->merge_symbols($obj, '100.MISSING'); $sym = $sym_file->lookup_symbol('symbol@Base', ['libinternal.so.0']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol@Base', minver => '1.0', dep_id => 0, deprecated => 0, ), 'symbol unaffected while including internal symbols via symbol tag'); $sym = $sym_file->lookup_symbol('.gomp_critical_user_foo@Base', ['libinternal.so.0']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => '.gomp_critical_user_foo@Base', minver => '2.0', dep_id => 0, deprecated => 0, tags => { 'allow-internal' => undef, }, tagorder => [ 'allow-internal' ], ), 'internal gomp symbol included via symbol tag'); $sym = $sym_file->lookup_symbol('__aeabi_lcmp@GCC_3.0', ['libinternal.so.0']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => '__aeabi_lcmp@GCC_3.0', minver => '3.0', dep_id => 0, deprecated => 0, tags => { 'allow-internal' => undef, }, tagorder => [ 'allow-internal' ], ), 'internal known aeabi symbol included via symbol tag'); $sym = $sym_file->lookup_symbol('__aeabi_unknown@GCC_4.0', ['libinternal.so.0']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => '__aeabi_unknown@GCC_4.0', minver => '4.0', dep_id => 0, deprecated => 0, tags => { 'allow-internal' => undef, }, tagorder => [ 'allow-internal' ], ), 'internal unknown aeabi symbol omitted via symbol tag'); # Include internal symbols using the Allow-Internal-Symbol-Groups field $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.internal-allow-groups"); $sym_file->merge_symbols($obj, '100.MISSING'); $sym = $sym_file->lookup_symbol('symbol@Base', ['libinternal.so.0']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol@Base', minver => '1.0', dep_id => 0, deprecated => 0, ), 'symbol unaffected w/o including internal symbols'); $sym = $sym_file->lookup_symbol('.gomp_critical_user_foo@Base', ['libinternal.so.0']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => '.gomp_critical_user_foo@Base', minver => '2.0', dep_id => 0, deprecated => 0, ), 'internal gomp symbol included via library field'); $sym = $sym_file->lookup_symbol('__aeabi_lcmp@GCC_3.0', ['libinternal.so.0']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => '__aeabi_lcmp@GCC_3.0', minver => '3.0', dep_id => 0, deprecated => 0, ), 'internal known aeabi symbol included via library field'); $sym = $sym_file->lookup_symbol('__aeabi_unknown@GCC_4.0', ['libinternal.so.0']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => '__aeabi_unknown@GCC_4.0', minver => '4.0', dep_id => 0, deprecated => 0, ), 'internal unknown aeabi symbol included via library field'); # Test include mechanism of SymbolFile $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.include-1"); $sym = $sym_file->lookup_symbol('symbol_before@Base', ['libfake.so.1']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol_before@Base', minver => '0.9', dep_id => 0, deprecated => 0, ), 'symbol before include not lost'); $sym = $sym_file->lookup_symbol('symbol_after@Base', ['libfake.so.1']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol_after@Base', minver => '1.1', dep_id => 0, deprecated => 0, ), 'symbol after include not lost'); $sym = $sym_file->lookup_symbol('symbol1_fake1@Base', ['libfake.so.1']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol1_fake1@Base', minver => '1.0', dep_id => 0, deprecated => 0, ), 'overrides order with #include'); $sym = $sym_file->lookup_symbol('symbol3_fake1@Base', ['libfake.so.1']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol3_fake1@Base', minver => '0', dep_id => 0, deprecated => 0, ), 'overrides order with #include'); is($sym_file->get_smallest_version('libfake.so.1'), '0', 'get_smallest_version with null version'); $sym = $sym_file->lookup_symbol('symbol_in_libdivert@Base', ['libdivert.so.1']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol_in_libdivert@Base', minver => '1.0~beta1', dep_id => 0, deprecated => 0, ), '#include can change current object'); $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.include-2"); $sym = $sym_file->lookup_symbol('symbol1_fake2@Base', ['libfake.so.1']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol1_fake2@Base', minver => '1.0', dep_id => 1, deprecated => 0, ), 'overrides order with circular #include'); is($sym_file->get_smallest_version('libfake.so.1'), '1.0', 'get_smallest_version'); my $io_data; my $io; # Check dump output open $io, '>', \$io_data or die "cannot open io string\n"; $sym_file->output($io, package => 'libfake1'); is($io_data, 'libfake.so.1 libfake1 #MINVER# | libvirtualfake * Build-Depends-Package: libfake-dev symbol1_fake2@Base 1.0 1 symbol2_fake2@Base 1.0 symbol3_fake2@Base 1.1 ', "Dump of $datadir/symbols.include-2"); # Check parsing of objdump output on ia64 (local symbols # without versions and with visibility attribute) $obj = load_objdump_obj('glib-ia64'); ok($obj->is_public_library(), 'glib-ia64 is a public library'); ok(!$obj->is_executable(), 'glib-ia64 is not an executable'); $sym = $obj->get_symbol('IA__g_free'); is_deeply($sym, { name => 'IA__g_free', version => '', soname => 'libglib-2.0.so.0', objid => 'libglib-2.0.so.0', section => '.text', dynamic => 1, debug => '', type => 'F', weak => '', local => 1, global => '', visibility => 'hidden', hidden => '', defined => 1, }, 'symbol with visibility without version'); # Check parsing of objdump output when symbol names contain spaces $obj = load_objdump_obj('spacesyms'); sub check_spacesym { my ($name, $version, $visibility) = @_; $visibility //= ''; $sym = $obj->get_symbol($name . "@" . $version); is_deeply($sym, { name => $name, version => $version, soname => 'libspacesyms.so.1', objid => 'libspacesyms.so.1', section => '.text', dynamic => 1, debug => '', type => 'F', weak => '', local => '', global => 1, visibility => $visibility, hidden => '', defined => 1, }, $name); ok(defined $obj->{dynrelocs}{$name . "@@" . $version}, "dynreloc found for $name"); } check_spacesym('symdefaultvernospacedefault', 'Base'); check_spacesym('symdefaultvernospaceprotected', 'Base', 'protected'); check_spacesym('symlongvernospacedefault', 'VERY_LONG_VERSION_1'); check_spacesym('symlongvernospaceprotected', 'VERY_LONG_VERSION_1', 'protected'); check_spacesym('symshortvernospacedefault', 'V1'); check_spacesym('symshortvernospaceprotected', 'V1', 'protected'); check_spacesym('symdefaultverSPA CEdefault', 'Base'); check_spacesym('symdefaultverSPA CEprotected', 'Base', 'protected'); check_spacesym('symlongverSPA CEdefault', 'VERY_LONG_VERSION_1'); check_spacesym('symlongverSPA CEprotected', 'VERY_LONG_VERSION_1', 'protected'); check_spacesym('symshortverSPA CEdefault', 'V1'); check_spacesym('symshortverSPA CEprotected', 'V1', 'protected'); ####### Test symbol tagging support ###### # Parsing/dumping # Template mode $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/basictags.symbols", arch => 'amd64'); save_load_test($sym_file, 'template save -> load', template_mode => 1); # Dumping in non-template mode (amd64) (test for arch tags) open $io, '>', \$io_data or die "cannot open io string\n"; $sym_file->output($io); is($io_data, 'libbasictags.so.1 libbasictags1 #MINVER# | libbasictags1 (>= 1.1) symbol11_optional@Base 1.1 1 symbol21_amd64@Base 2.1 symbol25_64@Base 2.5 symbol26_little@Base 2.6 symbol31_randomtag@Base 3.1 symbol51_untagged@Base 5.1 ', 'template vs. non-template on amd64'); # Dumping in non-template mode (mips) (test for arch tags) open $io, '>', \$io_data or die "cannot open io string\n"; $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/basictags.symbols", arch => 'mips'); $sym_file->output($io); is($io_data, 'libbasictags.so.1 libbasictags1 #MINVER# | libbasictags1 (>= 1.1) symbol11_optional@Base 1.1 1 symbol23_mips@Base 2.3 symbol24_32@Base 2.4 symbol27_big@Base 2.7 symbol31_randomtag@Base 3.1 symbol42_mips_and_optional@Base 4.2 symbol51_untagged@Base 5.1 ', 'template vs. non-template on mips'); # Dumping in non-template mode (i386) (test for arch tags) open $io, '>', \$io_data or die "cannot open io string\n"; $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/basictags.symbols", arch => 'i386'); $sym_file_dup = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/basictags.symbols", arch => 'i386'); $sym_file->output($io); is($io_data, 'libbasictags.so.1 libbasictags1 #MINVER# | libbasictags1 (>= 1.1) symbol11_optional@Base 1.1 1 symbol22_i386@Base 2.2 symbol24_32@Base 2.4 symbol26_little@Base 2.6 symbol28_little_32@Base 2.8 symbol31_randomtag@Base 3.1 symbol41_i386_and_optional@Base 4.1 symbol51_untagged@Base 5.1 ', 'template vs. non-template on i386'); ok(defined $sym_file->{objects}{'libbasictags.so.1'}{syms}{'symbol21_amd64@Base'}, 'syms keys are symbol names without quotes'); # Preload objdumps my $tags_obj_i386 = load_objdump_obj('basictags-i386'); $sym_file->merge_symbols($tags_obj_i386, '100.MISSING'); is_deeply($sym_file, $sym_file_dup, 'is objdump.basictags-i386 and basictags.symbols in sync'); my $tags_obj_amd64 = load_objdump_obj('basictags-amd64'); # Merge/get_{new,lost} tests for optional tag: # - disappeared my $symbol11 = $tags_obj_i386->get_symbol('symbol11_optional@Base'); delete $tags_obj_i386->{dynsyms}{'symbol11_optional@Base'}; $sym_file->merge_symbols($tags_obj_i386, '100.MISSING'); $sym = $sym_file->lookup_symbol('symbol11_optional@Base', ['libbasictags.so.1'], 1); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol11_optional@Base', symbol_templ => 'symbol11_optional@Base', minver => '1.1', dep_id => 1, deprecated => '100.MISSING', tags => { optional => undef, }, tagorder => [ 'optional' ], ), 'disappeared optional symbol gets deprecated'); $sym_file->merge_symbols($tags_obj_i386, '101.MISSING'); $sym = $sym_file->lookup_symbol('symbol11_optional@Base', ['libbasictags.so.1'], 1); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol11_optional@Base', symbol_templ => 'symbol11_optional@Base', minver => '1.1', dep_id => 1, deprecated => '101.MISSING', tags => { optional => undef, }, tagorder => [ 'optional' ], ), 'deprecated text of MISSING optional symbol gets rebumped each merge'); is(scalar($sym_file->get_lost_symbols($sym_file_dup)), 0, 'missing optional symbol is not LOST'); # - reappeared (undeprecate, minver should be 1.1, not 100.MISSED) $tags_obj_i386->add_dynamic_symbol($symbol11); $sym_file->merge_symbols($tags_obj_i386, '100.MISSING'); $sym = $sym_file->lookup_symbol('symbol11_optional@Base', ['libbasictags.so.1']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol11_optional@Base', symbol_templ => 'symbol11_optional@Base', minver => '1.1', dep_id => 1, deprecated => 0, tags => { optional => undef, }, tagorder => [ 'optional' ], ), 'reappeared optional symbol gets undeprecated + minver'); is(scalar($sym_file->get_lost_symbols($sym_file_dup) + $sym_file->get_new_symbols($sym_file_dup)), 0, 'reappeared optional symbol: neither NEW nor LOST'); # Merge/get_{new,lost} tests for arch tag: # - arch specific appears on wrong arch: 'arch' tag should be removed my $symbol21 = $tags_obj_amd64->get_symbol('symbol21_amd64@Base'); $tags_obj_i386->add_dynamic_symbol($symbol21); $sym_file->merge_symbols($tags_obj_i386, '100.MISSING'); $sym = $sym_file->lookup_symbol('symbol21_amd64@Base', ['libbasictags.so.1']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol21_amd64@Base', symbol_templ => 'symbol21_amd64@Base', symbol_quoted => "'", minver => '2.1', dep_id => 0, deprecated => 0, ), 'symbol appears on foreign arch, arch tag should be removed'); @tmp = map { $_->{symbol}->get_symbolname() } $sym_file->get_new_symbols($sym_file_dup); is_deeply(\@tmp, [ 'symbol21_amd64@Base' ], 'symbol from foreign arch is NEW'); is($sym->get_symbolspec(1), ' symbol21_amd64@Base 2.1', 'no tags => no quotes in the symbol name'); # - arch specific symbol disappears delete $tags_obj_i386->{dynsyms}{'symbol22_i386@Base'}; delete $tags_obj_i386->{dynsyms}{'symbol24_32@Base'}; delete $tags_obj_i386->{dynsyms}{'symbol26_little@Base'}; delete $tags_obj_i386->{dynsyms}{'symbol28_little_32@Base'}; delete $tags_obj_i386->{dynsyms}{'symbol41_i386_and_optional@Base'}; $sym_file->merge_symbols($tags_obj_i386, '100.MISSING'); $sym = $sym_file->lookup_symbol('symbol22_i386@Base', ['libbasictags.so.1'], 1); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol22_i386@Base', symbol_templ => 'symbol22_i386@Base', minver => '2.2', dep_id => 0, deprecated => '100.MISSING', tags => { arch => '!amd64 !ia64 !mips', }, tagorder => [ 'arch' ], ), 'disappeared arch specific symbol gets deprecated'); $sym = $sym_file->lookup_symbol('symbol24_32@Base', ['libbasictags.so.1'], 1); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol24_32@Base', symbol_templ => 'symbol24_32@Base', minver => '2.4', dep_id => 0, deprecated => '100.MISSING', tags => { 'arch-bits' => '32', }, tagorder => [ 'arch-bits' ], ), 'disappeared arch bits specific symbol gets deprecated'); $sym = $sym_file->lookup_symbol('symbol26_little@Base', ['libbasictags.so.1'], 1); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol26_little@Base', symbol_templ => 'symbol26_little@Base', minver => '2.6', dep_id => 0, deprecated => '100.MISSING', tags => { 'arch-endian' => 'little', }, tagorder => [ 'arch-endian' ], ), 'disappeared arch endian specific symbol gets deprecated'); $sym = $sym_file->lookup_symbol('symbol28_little_32@Base', ['libbasictags.so.1'], 1); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol28_little_32@Base', symbol_templ => 'symbol28_little_32@Base', minver => '2.8', dep_id => 0, deprecated => '100.MISSING', tags => { 'arch-bits' => '32', 'arch-endian' => 'little', }, tagorder => [ 'arch-bits', 'arch-endian' ], ), 'disappeared arch bits and endian specific symbol gets deprecated'); $sym = $sym_file->lookup_symbol('symbol41_i386_and_optional@Base', ['libbasictags.so.1'], 1); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol41_i386_and_optional@Base', symbol_templ => 'symbol41_i386_and_optional@Base', symbol_quoted => '"', minver => '4.1', dep_id => 0, deprecated => '100.MISSING', tags => { arch => 'i386', optional => 'reason', }, tagorder => [ 'arch', 'optional' ], ), 'disappeared optional arch specific symbol gets deprecated'); @tmp = sort map { $_->{symbol}->get_symbolname() } $sym_file->get_lost_symbols($sym_file_dup); is_deeply(\@tmp, [ 'symbol22_i386@Base', 'symbol24_32@Base', 'symbol26_little@Base', 'symbol28_little_32@Base', ], "missing arch specific is LOST, but optional arch specific isn't"); # Tests for tagged #includes $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.include-3", arch => 'i386'); $sym = $sym_file->lookup_symbol('symbol2_fake1@Base', ['libbasictags.so.2']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol2_fake1@Base', minver => '1.0', tags => { optional => undef, 'random tag' => 'random value', }, tagorder => [ 'optional', 'random tag' ], ), 'symbols from #included file inherits tags'); $sym = $sym_file->lookup_symbol('symbol41_i386_and_optional@Base', ['libbasictags.so.1']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol41_i386_and_optional@Base', symbol_templ => 'symbol41_i386_and_optional@Base', symbol_quoted => '"', minver => '4.1', tags => { optional => 'reason', t => 'v', arch => 'i386', }, tagorder => [ 'optional', 't', 'arch' ], ), 'symbols in #included file can override tag values'); $sym = $sym_file->lookup_symbol('symbol51_untagged@Base', ['libbasictags.so.1']); is_deeply($sym, Dpkg::Shlibs::Symbol->new( symbol => 'symbol51_untagged@Base', minver => '5.1', tags => { optional => 'from parent', t => 'v', }, tagorder => [ 'optional', 't' ], ), 'symbols are properly cloned when #including'); # Test Symbol::clone() $sym = Dpkg::Shlibs::Symbol->new( symbol => 'foobar', testfield => 1, teststruct => { foo => 1, }, ); $tmp = $sym->clone(); $tmp->{teststruct}{foo} = 3; $tmp->{testfield} = 3; is($sym->{teststruct}{foo}, 1, 'original field "foo" not changed'); is($sym->{testfield}, 1, 'original field "testfield" not changed'); ############ Test symbol patterns ########### SKIP: { skip 'c++filt not available', 41 if not can_run('c++filt'); sub load_patterns_obj { $obj = load_objdump_obj('patterns'); return $obj; } sub load_patterns_symbols { $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/patterns.symbols"); return $sym_file; } load_patterns_obj(); $sym_file_dup = load_patterns_symbols(); load_patterns_symbols(); save_load_test($sym_file, 'save -> load test of patterns template', template_mode => 1); isnt($sym_file->get_patterns('libpatterns.so.1'), 0, 'patterns.symbols has patterns'); $sym_file->merge_symbols($obj, '100.MISSING'); @tmp = map { $_->get_symbolname() } $sym_file->get_lost_symbols($sym_file_dup); is_deeply(\@tmp, [], 'no LOST symbols if all patterns matched.'); @tmp = map { $_->get_symbolname() } $sym_file->get_new_symbols($sym_file_dup); is_deeply(\@tmp, [], 'no NEW symbols if all patterns matched.'); # Pattern resolution order: aliases (c++, symver), generic $sym = $sym_file->lookup_symbol('SYMVER_1@SYMVER_1', 'libpatterns.so.1'); is($sym->{minver}, '1', 'specific SYMVER_1 symbol'); $sym = $sym_file->lookup_symbol('_ZN3NSB6Symver14symver_method1Ev@SYMVER_1', 'libpatterns.so.1'); is($sym->{minver}, '1.method1', 'specific symbol preferred over pattern'); $sym = $sym_file->lookup_symbol('_ZN3NSB6Symver14symver_method2Ev@SYMVER_1', 'libpatterns.so.1'); is($sym->{minver}, '1.method2', 'c++ alias pattern preferred over generic pattern'); is($sym->get_pattern()->get_symbolname(), 'NSB::Symver::symver_method2()@SYMVER_1', 'c++ alias pattern preferred over generic pattern, on demangled name'); $sym = $sym_file->lookup_symbol('_ZN3NSB6SymverD1Ev@SYMVER_1', 'libpatterns.so.1'); is($sym->{minver}, '1.generic', 'generic (c++ & symver) pattern covers the rest (destructor)'); ok($sym->get_pattern()->equals($sym_file->create_symbol('(c++|symver)SYMVER_1 1.generic')), 'generic (c++ & symver) pattern covers the rest (destructor), compared'); # Test old style wildcard support load_patterns_symbols(); $sym = $sym_file->create_symbol('*@SYMVEROPT_2 2'); ok($sym->is_optional(), 'Old style wildcard is optional'); is($sym->get_alias_type(), 'symver', 'old style wildcard is a symver pattern'); is($sym->get_symbolname(), 'SYMVEROPT_2', 'wildcard pattern got renamed'); $pat = $sym_file->lookup_pattern('(symver|optional)SYMVEROPT_2', 'libpatterns.so.1'); $sym->{symbol_templ} = $pat->{symbol_templ}; is_deeply($pat, $sym, 'old style wildcard is the same as (symver|optional)'); # Get rid of all SymverOptional symbols foreach my $tmp (keys %{$obj->{dynsyms}}) { delete $obj->{dynsyms}{$tmp} if $tmp =~ /SymverOptional/; } $sym_file->merge_symbols($obj, '100.MISSING'); is_deeply([ map { $_->get_symbolname() } $pat->get_pattern_matches() ], [], 'old style wildcard matches nothing.'); is($pat->{deprecated}, '100.MISSING', 'old style wildcard gets deprecated.'); @tmp = map { $_->{symbol}->get_symbolname() } $sym_file->get_lost_symbols($sym_file_dup); is_deeply(\@tmp, [], 'but old style wildcard is not LOST.'); # 'Internal' pattern covers all internal symbols load_patterns_obj(); @tmp = grep { $_->get_symbolname() =~ /Internal/ } $sym_file->get_symbols('libpatterns.so.1'); $sym = $sym_file->create_symbol('(regex|c++)^_Z(T[ISV])?N3NSA6ClassA8Internal.*@Base$ 1.internal'); $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1'); is_deeply([ sort $pat->get_pattern_matches() ], [ sort @tmp ], 'Pattern covers all internal symbols'); is($tmp[0]->{minver}, '1.internal', 'internal pattern covers first symbol'); # Lookup private pattern my @private_symnames = sort qw( _ZN3NSA6ClassA7Private11privmethod1Ei@Base _ZN3NSA6ClassA7Private11privmethod2Ei@Base _ZN3NSA6ClassA7PrivateC1Ev@Base _ZN3NSA6ClassA7PrivateC2Ev@Base _ZN3NSA6ClassA7PrivateD0Ev@Base _ZN3NSA6ClassA7PrivateD1Ev@Base _ZN3NSA6ClassA7PrivateD2Ev@Base _ZTIN3NSA6ClassA7PrivateE@Base _ZTSN3NSA6ClassA7PrivateE@Base _ZTVN3NSA6ClassA7PrivateE@Base ); $sym = $sym_file->create_symbol('(c++|regex|optional)NSA::ClassA::Private(::.*)?@Base 1'); $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1'); isnt($pat, undef, 'pattern for private class has been found'); is_deeply([ sort map { $_->get_symbolname() } $pat->get_pattern_matches() ], \@private_symnames, 'private pattern matched expected symbols'); ok(($pat->get_pattern_matches())[0]->is_optional(), 'private symbol is optional like its pattern'); ok($sym_file->lookup_symbol(($pat->get_pattern_matches())[0], 'libpatterns.so.1'), 'lookup_symbol() finds symbols matched by pattern (after merge)'); # Get rid of a private symbol, it should not be lost delete $obj->{dynsyms}{$private_symnames[0]}; load_patterns_symbols(); $sym_file->merge_symbols($obj, '100.MISSING'); $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1'); @tmp = map { $_->{symbol}->get_symbolname() } $sym_file->get_lost_symbols($sym_file_dup); is_deeply(\@tmp, [], 'no LOST symbols when got rid of patterned optional symbol.'); ok(!$pat->{deprecated}, 'there are still matches, pattern is not deprecated.'); # Get rid of all private symbols, the pattern should be deprecated. foreach my $tmp (@private_symnames) { delete $obj->{dynsyms}{$tmp}; } load_patterns_symbols(); $sym_file->merge_symbols($obj, '100.MISSING'); $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1', 1); @tmp = $sym_file->get_lost_symbols($sym_file_dup); is_deeply(\@tmp, [ ], 'All private symbols gone, but pattern is not LOST because it is optional.'); is($pat->{deprecated}, '100.MISSING', 'All private symbols gone - pattern deprecated.'); # Internal symbols. All covered by the pattern? @tmp = grep { $_->get_symbolname() =~ /Internal/ } values %{$sym_file->{objects}{'libpatterns.so.1'}{syms}}; $sym = $sym_file->create_symbol('(regex|c++)^_Z(T[ISV])?N3NSA6ClassA8Internal.*@Base$ 1.internal'); $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1'); is_deeply([ sort $pat->get_pattern_matches() ], [ sort @tmp ], 'Pattern covers all internal symbols'); is($tmp[0]->{minver}, '1.internal', 'internal pattern covers first symbol'); # Delete matches of the non-optional pattern $sym = $sym_file->create_symbol('(c++)"non-virtual thunk to NSB::ClassD::generate_vt(char const*) const@Base" 1'); $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1'); isnt($pat, undef, 'lookup_pattern() finds alias-based pattern'); is(scalar($pat->get_pattern_matches()), 2, 'two matches for the generate_vt pattern'); foreach my $tmp ($pat->get_pattern_matches()) { delete $obj->{dynsyms}{$tmp->get_symbolname()}; } load_patterns_symbols(); $sym_file->merge_symbols($obj, '100.MISSING'); $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1', 1); @tmp = map { scalar $sym_file->lookup_pattern($_->{symbol}, 'libpatterns.so.1', 1) } $sym_file->get_lost_symbols($sym_file_dup); is_deeply(\@tmp, [ $pat ], 'No matches - generate_vt() pattern is LOST.'); is($pat->{deprecated}, '100.MISSING', 'No matches - generate_vt() pattern is deprecated.'); # Pattern undeprecation when matches are discovered load_patterns_obj(); load_patterns_symbols(); $pat = $sym_file_dup->lookup_pattern($sym, 'libpatterns.so.1'); $pat->{deprecated} = '0.1-1'; $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1'); $pat->{deprecated} = '0.1-1'; $sym_file->merge_symbols($obj, '100.FOUND'); ok(! $pat->{deprecated}, 'Previously deprecated pattern with matches got undeprecated'); is($pat->{minver}, '100.FOUND', 'Previously deprecated pattern with matches got minver bumped'); @tmp = map { $_->{symbol}->get_symbolspec(1) } $sym_file->get_new_symbols($sym_file_dup); is_deeply(\@tmp, [ $pat->get_symbolspec(1) ], 'Previously deprecated pattern with matches is NEW. Matches themselves are not NEW.'); foreach my $sym ($pat->get_pattern_matches()) { ok(!$sym->{deprecated}, $sym->get_symbolname() . ': not deprecated'); is($sym->{minver}, '100.FOUND', $sym->get_symbolname() . ': version bumped'); } }