Dpkg_Shlibs.t 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781
  1. #!/usr/bin/perl
  2. #
  3. # This program is free software; you can redistribute it and/or modify
  4. # it under the terms of the GNU General Public License as published by
  5. # the Free Software Foundation; either version 2 of the License, or
  6. # (at your option) any later version.
  7. #
  8. # This program is distributed in the hope that it will be useful,
  9. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. # GNU General Public License for more details.
  12. #
  13. # You should have received a copy of the GNU General Public License
  14. # along with this program. If not, see <https://www.gnu.org/licenses/>.
  15. use strict;
  16. use warnings;
  17. use Test::More;
  18. use Test::Dpkg qw(:needs :paths);
  19. use Cwd;
  20. test_needs_module('IO::String');
  21. plan tests => 148;
  22. use Dpkg::Path qw(find_command);
  23. use_ok('Dpkg::Shlibs');
  24. my $tmp;
  25. my @tmp;
  26. my %tmp;
  27. my $datadir = test_get_data_path('t/Dpkg_Shlibs');
  28. my @librarypaths;
  29. {
  30. # XXX: Keep as long as we support the deprecated LD_LIBRARY_PATH.
  31. local $ENV{LD_LIBRARY_PATH} = '/test-env';
  32. Dpkg::Shlibs::add_library_dir('/test-a');
  33. @librarypaths = Dpkg::Shlibs::get_library_paths();
  34. is($librarypaths[0], '/test-a', 'add_library_dir() does not get lost');
  35. Dpkg::Shlibs::add_library_dir('/test-b');
  36. @librarypaths = Dpkg::Shlibs::get_library_paths();
  37. is_deeply([ @librarypaths[0, 1] ] , [ '/test-a', '/test-b' ],
  38. 'add_library_dir() prepends');
  39. }
  40. Dpkg::Shlibs::blank_library_paths();
  41. # We want relative paths inside the ld.so.conf fragments to work, and $srcdir
  42. # is usually a relative path, so let's temporarily switch directory.
  43. # XXX: An alternative would be to make parse_ldso_conf relative path aware.
  44. my $cwd = cwd();
  45. test_needs_srcdir_switch();
  46. Dpkg::Shlibs::parse_ldso_conf('t/Dpkg_Shlibs/ld.so.conf');
  47. chdir($cwd);
  48. @librarypaths = Dpkg::Shlibs::get_library_paths();
  49. is_deeply(\@librarypaths,
  50. [ qw(/nonexistant32 /nonexistant/lib64
  51. /usr/local/lib /nonexistant/lib128) ], 'parsed library paths');
  52. use_ok('Dpkg::Shlibs::Objdump');
  53. my $obj = Dpkg::Shlibs::Objdump::Object->new;
  54. open my $objdump, '<', "$datadir/objdump.dbd-pg"
  55. or die "$datadir/objdump.dbd-pg: $!";
  56. $obj->parse_objdump_output($objdump);
  57. close $objdump;
  58. ok(!$obj->is_public_library(), 'Pg.so is not a public library');
  59. ok(!$obj->is_executable(), 'Pg.so is not an executable');
  60. open $objdump, '<', "$datadir/objdump.ls"
  61. or die "$datadir/objdump.ls: $!";
  62. $obj->reset();
  63. $obj->parse_objdump_output($objdump);
  64. close $objdump;
  65. ok(!$obj->is_public_library(), 'ls is not a public library');
  66. ok($obj->is_executable(), 'ls is an executable');
  67. my $sym = $obj->get_symbol('optarg@GLIBC_2.0');
  68. ok($sym, 'optarg@GLIBC_2.0 exists');
  69. ok(!$sym->{defined}, 'R_*_COPY relocations are taken into account');
  70. open $objdump, '<', "$datadir/objdump.space"
  71. or die "$datadir/objdump.space: $!";
  72. $obj->reset();
  73. $obj->parse_objdump_output($objdump);
  74. close $objdump;
  75. # Non-regression test for #506139
  76. $sym = $obj->get_symbol('singlespace');
  77. ok($sym, 'version less symbol separated by a single space are correctly parsed');
  78. open $objdump, '<', "$datadir/objdump.libc6-2.6"
  79. or die "$datadir/objdump.libc6-2.6: $!";
  80. $obj->reset();
  81. $obj->parse_objdump_output($objdump);
  82. close $objdump;
  83. ok($obj->is_public_library(), 'libc6 is a public library');
  84. ok(!$obj->is_executable(), 'libc6 is not an executable');
  85. is($obj->{SONAME}, 'libc.so.6', 'SONAME');
  86. is($obj->{HASH}, '0x13d99c', 'HASH');
  87. is($obj->{GNU_HASH}, '0x194', 'GNU_HASH');
  88. is($obj->{format}, 'elf32-i386', 'format');
  89. is_deeply($obj->{flags}, { DYNAMIC => 1, HAS_SYMS => 1, D_PAGED => 1 }, 'flags');
  90. is_deeply($obj->{NEEDED}, [ 'ld-linux.so.2' ], 'NEEDED');
  91. is_deeply([ $obj->get_needed_libraries ], [ 'ld-linux.so.2' ], 'NEEDED');
  92. $sym = $obj->get_symbol('_sys_nerr@GLIBC_2.3');
  93. is_deeply( $sym, { name => '_sys_nerr', version => 'GLIBC_2.3',
  94. soname => 'libc.so.6', objid => 'libc.so.6',
  95. section => '.rodata', dynamic => 1,
  96. debug => '', type => 'O', weak => '',
  97. local => '', global => 1, visibility => '',
  98. hidden => 1, defined => 1 }, 'Symbol' );
  99. $sym = $obj->get_symbol('_IO_stdin_used');
  100. is_deeply( $sym, { name => '_IO_stdin_used', version => '',
  101. soname => 'libc.so.6', objid => 'libc.so.6',
  102. section => '*UND*', dynamic => 1,
  103. debug => '', type => ' ', weak => 1,
  104. local => '', global => '', visibility => '',
  105. hidden => '', defined => '' }, 'Symbol 2' );
  106. my @syms = $obj->get_exported_dynamic_symbols;
  107. is( scalar @syms, 2231, 'defined && dynamic' );
  108. @syms = $obj->get_undefined_dynamic_symbols;
  109. is( scalar @syms, 9, 'undefined && dynamic' );
  110. my $obj_old = Dpkg::Shlibs::Objdump::Object->new;
  111. open $objdump, '<', "$datadir/objdump.libc6-2.3"
  112. or die "$datadir/objdump.libc6-2.3: $!";
  113. $obj_old->parse_objdump_output($objdump);
  114. close $objdump;
  115. use_ok('Dpkg::Shlibs::SymbolFile');
  116. use_ok('Dpkg::Shlibs::Symbol');
  117. my $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbol_file.tmp");
  118. my $sym_file_dup = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbol_file.tmp");
  119. my $sym_file_old = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbol_file.tmp");
  120. $sym_file->merge_symbols($obj_old, '2.3.6.ds1-13');
  121. $sym_file_old->merge_symbols($obj_old, '2.3.6.ds1-13');
  122. ok( $sym_file->has_object('libc.so.6'), 'SONAME in sym file' );
  123. $sym_file->merge_symbols($obj, '2.6-1');
  124. ok( $sym_file->get_new_symbols($sym_file_old), 'has new symbols' );
  125. ok( $sym_file_old->get_lost_symbols($sym_file), 'has lost symbols' );
  126. is( $sym_file_old->lookup_symbol('__bss_start@Base', ['libc.so.6']),
  127. undef, 'internal symbols are blacklisted' );
  128. %tmp = $sym_file->lookup_symbol('_errno@GLIBC_2.0', ['libc.so.6'], 1);
  129. isa_ok($tmp{symbol}, 'Dpkg::Shlibs::Symbol');
  130. is_deeply(\%tmp, { symbol => Dpkg::Shlibs::Symbol->new(symbol => '_errno@GLIBC_2.0',
  131. minver => '2.3.6.ds1-13', dep_id => 0,
  132. deprecated => '2.6-1'), soname => 'libc.so.6' },
  133. 'deprecated symbol');
  134. # Wildcard test
  135. my $pat = $sym_file_old->create_symbol('*@GLIBC_PRIVATE 2.3.6.wildcard');
  136. $sym_file_old->add_symbol($pat, 'libc.so.6');
  137. $sym_file_old->merge_symbols($obj, '2.6-1');
  138. $sym = $sym_file_old->lookup_symbol('__nss_services_lookup@GLIBC_PRIVATE', 'libc.so.6');
  139. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => '__nss_services_lookup@GLIBC_PRIVATE',
  140. minver => '2.3.6.wildcard', dep_id => 0, deprecated => 0,
  141. tags => { symver => undef, optional => undef },
  142. tagorder => [ 'symver', 'optional' ],
  143. matching_pattern => $pat ), 'wildcarded symbol');
  144. # Save -> Load test
  145. use File::Temp;
  146. use File::Basename qw(basename);
  147. sub save_load_test {
  148. my ($symfile, $comment, @opts) = @_;
  149. my $save_file = File::Temp->new();
  150. $symfile->save($save_file->filename, @opts);
  151. my $dup = Dpkg::Shlibs::SymbolFile->new(file => $save_file->filename);
  152. # Force sync of non-stored attributes
  153. $dup->{file} = $symfile->{file};
  154. $dup->{arch} = $symfile->{arch};
  155. is_deeply($dup, $symfile, $comment);
  156. if (-f $symfile->{file}) {
  157. is(system('diff', '-u', $symfile->{file}, $save_file->filename), 0,
  158. basename($symfile->{file}) . ' dumped identical');
  159. }
  160. }
  161. save_load_test( $sym_file, 'save -> load' );
  162. # Test ignoring blacklisted symbols
  163. open $objdump, '<', "$datadir/objdump.blacklisted"
  164. or die "objdump.blacklisted: $!";
  165. $obj->reset();
  166. $obj->parse_objdump_output($objdump);
  167. close $objdump;
  168. # Do not ignore any blacklist
  169. $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.blacklist-filter");
  170. $sym_file->merge_symbols($obj, '100.MISSING');
  171. $sym = $sym_file->lookup_symbol('symbol@Base', ['libblacklisted.so.0']);
  172. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol@Base',
  173. minver => '1.0', dep_id => 0, deprecated => 0),
  174. 'symbol unaffected w/o ignoring blacklists');
  175. $sym = $sym_file->lookup_symbol('.gomp_critical_user_foo@Base', ['libblacklisted.so.0']);
  176. is($sym, undef, 'gomp symbol omitted while filtering on blacklists');
  177. $sym = $sym_file->lookup_symbol('__aeabi_lcmp@GCC_3.0', ['libblacklisted.so.0']);
  178. is($sym, undef, 'known aeabi symbol omitted while filtering on blacklists');
  179. $sym = $sym_file->lookup_symbol('__aeabi_unknown@GCC_4.0', ['libblacklisted.so.0']);
  180. is($sym, undef, 'unknown aeabi symbol omitted while filtering on blacklists');
  181. # Ignore blacklists using the ignore-blacklists symbol tag
  182. $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.blacklist-ignore");
  183. $sym_file->merge_symbols($obj, '100.MISSING');
  184. $sym = $sym_file->lookup_symbol('symbol@Base', ['libblacklisted.so.0']);
  185. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol@Base',
  186. minver => '1.0', dep_id => 0, deprecated => 0),
  187. 'symbol unaffected while ignoring blacklists via symbol tag');
  188. $sym = $sym_file->lookup_symbol('.gomp_critical_user_foo@Base', ['libblacklisted.so.0']);
  189. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => '.gomp_critical_user_foo@Base',
  190. minver => '2.0', dep_id => 0, deprecated => 0,
  191. tags => { 'ignore-blacklist' => undef },
  192. tagorder => [ 'ignore-blacklist' ]),
  193. 'blacklisted gomp symbol included via symbol tag');
  194. $sym = $sym_file->lookup_symbol('__aeabi_lcmp@GCC_3.0', ['libblacklisted.so.0']);
  195. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => '__aeabi_lcmp@GCC_3.0',
  196. minver => '3.0', dep_id => 0, deprecated => 0,
  197. tags => { 'ignore-blacklist' => undef },
  198. tagorder => [ 'ignore-blacklist' ]),
  199. 'blacklisted known aeabi symbol included via symbol tag');
  200. $sym = $sym_file->lookup_symbol('__aeabi_unknown@GCC_4.0', ['libblacklisted.so.0']);
  201. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => '__aeabi_unknown@GCC_4.0',
  202. minver => '4.0', dep_id => 0, deprecated => 0,
  203. tags => { 'ignore-blacklist' => undef },
  204. tagorder => [ 'ignore-blacklist' ]),
  205. 'blacklisted unknown aeabi symbol omitted via symbol tag');
  206. # Ignore blacklists using the Ignore-Blacklist-Groups field
  207. $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.blacklist-groups");
  208. $sym_file->merge_symbols($obj, '100.MISSING');
  209. $sym = $sym_file->lookup_symbol('symbol@Base', ['libblacklisted.so.0']);
  210. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol@Base',
  211. minver => '1.0', dep_id => 0, deprecated => 0),
  212. 'symbol unaffected w/o ignoring blacklists');
  213. $sym = $sym_file->lookup_symbol('.gomp_critical_user_foo@Base', ['libblacklisted.so.0']);
  214. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => '.gomp_critical_user_foo@Base',
  215. minver => '2.0', dep_id => 0, deprecated => 0),
  216. 'blacklisted gomp symbol included via library field');
  217. $sym = $sym_file->lookup_symbol('__aeabi_lcmp@GCC_3.0', ['libblacklisted.so.0']);
  218. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => '__aeabi_lcmp@GCC_3.0',
  219. minver => '3.0', dep_id => 0, deprecated => 0),
  220. 'blacklisted known aeabi symbol included via library field');
  221. $sym = $sym_file->lookup_symbol('__aeabi_unknown@GCC_4.0', ['libblacklisted.so.0']);
  222. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => '__aeabi_unknown@GCC_4.0',
  223. minver => '4.0', dep_id => 0, deprecated => 0),
  224. 'blacklisted unknown aeabi symbol included via library field');
  225. # Test include mechanism of SymbolFile
  226. $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.include-1");
  227. $sym = $sym_file->lookup_symbol('symbol_before@Base', ['libfake.so.1']);
  228. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol_before@Base',
  229. minver => '0.9', dep_id => 0, deprecated => 0),
  230. 'symbol before include not lost');
  231. $sym = $sym_file->lookup_symbol('symbol_after@Base', ['libfake.so.1']);
  232. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol_after@Base',
  233. minver => '1.1', dep_id => 0, deprecated => 0),
  234. 'symbol after include not lost');
  235. $sym = $sym_file->lookup_symbol('symbol1_fake1@Base', ['libfake.so.1']);
  236. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol1_fake1@Base',
  237. minver => '1.0', dep_id => 0, deprecated => 0),
  238. 'overrides order with #include');
  239. $sym = $sym_file->lookup_symbol('symbol3_fake1@Base', ['libfake.so.1']);
  240. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol3_fake1@Base',
  241. minver => '0', dep_id => 0, deprecated => 0),
  242. 'overrides order with #include');
  243. is($sym_file->get_smallest_version('libfake.so.1'), '0',
  244. 'get_smallest_version with null version');
  245. $sym = $sym_file->lookup_symbol('symbol_in_libdivert@Base', ['libdivert.so.1']);
  246. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol_in_libdivert@Base',
  247. minver => '1.0~beta1', dep_id => 0, deprecated => 0),
  248. '#include can change current object');
  249. $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.include-2");
  250. $sym = $sym_file->lookup_symbol('symbol1_fake2@Base', ['libfake.so.1']);
  251. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol1_fake2@Base',
  252. minver => '1.0', dep_id => 1, deprecated => 0),
  253. 'overrides order with circular #include');
  254. is($sym_file->get_smallest_version('libfake.so.1'), '1.0',
  255. 'get_smallest_version');
  256. # Check dump output
  257. my $io = IO::String->new();
  258. $sym_file->output($io, package => 'libfake1');
  259. is(${$io->string_ref()},
  260. 'libfake.so.1 libfake1 #MINVER#
  261. | libvirtualfake
  262. * Build-Depends-Package: libfake-dev
  263. symbol1_fake2@Base 1.0 1
  264. symbol2_fake2@Base 1.0
  265. symbol3_fake2@Base 1.1
  266. ', "Dump of $datadir/symbols.include-2");
  267. # Check parsing of objdump output on ia64 (local symbols
  268. # without versions and with visibility attribute)
  269. $obj = Dpkg::Shlibs::Objdump::Object->new;
  270. open $objdump, '<', "$datadir/objdump.glib-ia64"
  271. or die "$datadir/objdump.glib-ia64: $!";
  272. $obj->parse_objdump_output($objdump);
  273. close $objdump;
  274. $sym = $obj->get_symbol('IA__g_free');
  275. is_deeply( $sym, { name => 'IA__g_free', version => '',
  276. soname => 'libglib-2.0.so.0', objid => 'libglib-2.0.so.0',
  277. section => '.text', dynamic => 1,
  278. debug => '', type => 'F', weak => '',
  279. local => 1, global => '', visibility => 'hidden',
  280. hidden => '', defined => 1 },
  281. 'symbol with visibility without version' );
  282. # Check parsing of objdump output when symbol names contain spaces
  283. $obj = Dpkg::Shlibs::Objdump::Object->new;
  284. open $objdump, '<', "$datadir/objdump.spacesyms"
  285. or die "$datadir/objdump.spacesyms: $!";
  286. $obj->parse_objdump_output($objdump);
  287. close $objdump;
  288. sub check_spacesym {
  289. my ($name, $version, $visibility) = @_;
  290. $visibility //= '';
  291. $sym = $obj->get_symbol($name . "@" . $version);
  292. is_deeply($sym, { name => $name, version => $version,
  293. soname => 'libspacesyms.so.1',
  294. objid => 'libspacesyms.so.1',
  295. section => '.text', dynamic => 1,
  296. debug => '', type => 'F', weak => '',
  297. local => '', global => 1, visibility => $visibility,
  298. hidden => '', defined => 1 }, $name);
  299. ok(defined $obj->{dynrelocs}{$name}, "dynreloc found for $name");
  300. }
  301. check_spacesym('symdefaultvernospacedefault', 'Base');
  302. check_spacesym('symdefaultvernospaceprotected', 'Base', 'protected');
  303. check_spacesym('symlongvernospacedefault', 'VERY_LONG_VERSION_1');
  304. check_spacesym('symlongvernospaceprotected', 'VERY_LONG_VERSION_1', 'protected');
  305. check_spacesym('symshortvernospacedefault', 'V1');
  306. check_spacesym('symshortvernospaceprotected', 'V1', 'protected');
  307. check_spacesym('symdefaultverSPA CEdefault', 'Base');
  308. check_spacesym('symdefaultverSPA CEprotected', 'Base', 'protected');
  309. check_spacesym('symlongverSPA CEdefault', 'VERY_LONG_VERSION_1');
  310. check_spacesym('symlongverSPA CEprotected', 'VERY_LONG_VERSION_1', 'protected');
  311. check_spacesym('symshortverSPA CEdefault', 'V1');
  312. check_spacesym('symshortverSPA CEprotected', 'V1', 'protected');
  313. ####### Test symbol tagging support ######
  314. # Parsing/dumping
  315. # Template mode
  316. $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/basictags.symbols", arch => 'amd64');
  317. save_load_test($sym_file, 'template save -> load', template_mode => 1);
  318. # Dumping in non-template mode (amd64) (test for arch tags)
  319. $io = IO::String->new();
  320. $sym_file->output($io);
  321. is(${$io->string_ref()},
  322. 'libbasictags.so.1 libbasictags1 #MINVER#
  323. | libbasictags1 (>= 1.1)
  324. symbol11_optional@Base 1.1 1
  325. symbol21_amd64@Base 2.1
  326. symbol25_64@Base 2.5
  327. symbol26_little@Base 2.6
  328. symbol31_randomtag@Base 3.1
  329. symbol51_untagged@Base 5.1
  330. ', 'template vs. non-template on amd64');
  331. # Dumping in non-template mode (mips) (test for arch tags)
  332. $io = IO::String->new();
  333. $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/basictags.symbols", arch => 'mips');
  334. $sym_file->output($io);
  335. is(${$io->string_ref()},
  336. 'libbasictags.so.1 libbasictags1 #MINVER#
  337. | libbasictags1 (>= 1.1)
  338. symbol11_optional@Base 1.1 1
  339. symbol23_mips@Base 2.3
  340. symbol24_32@Base 2.4
  341. symbol27_big@Base 2.7
  342. symbol31_randomtag@Base 3.1
  343. symbol42_mips_and_optional@Base 4.2
  344. symbol51_untagged@Base 5.1
  345. ', 'template vs. non-template on mips');
  346. # Dumping in non-template mode (i386) (test for arch tags)
  347. $io = IO::String->new();
  348. $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/basictags.symbols", arch => 'i386');
  349. $sym_file_dup = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/basictags.symbols", arch => 'i386');
  350. $sym_file->output($io);
  351. is(${$io->string_ref()},
  352. 'libbasictags.so.1 libbasictags1 #MINVER#
  353. | libbasictags1 (>= 1.1)
  354. symbol11_optional@Base 1.1 1
  355. symbol22_i386@Base 2.2
  356. symbol24_32@Base 2.4
  357. symbol26_little@Base 2.6
  358. symbol28_little_32@Base 2.8
  359. symbol31_randomtag@Base 3.1
  360. symbol41_i386_and_optional@Base 4.1
  361. symbol51_untagged@Base 5.1
  362. ', 'template vs. non-template on i386');
  363. ok (defined $sym_file->{objects}{'libbasictags.so.1'}{syms}{'symbol21_amd64@Base'},
  364. 'syms keys are symbol names without quotes');
  365. # Preload objdumps
  366. my $tags_obj_i386 = Dpkg::Shlibs::Objdump::Object->new();
  367. open $objdump, '<', "$datadir/objdump.basictags-i386"
  368. or die "$datadir/objdump.basictags-i386: $!";
  369. $tags_obj_i386->parse_objdump_output($objdump);
  370. close $objdump;
  371. $sym_file->merge_symbols($tags_obj_i386, '100.MISSING');
  372. is_deeply($sym_file, $sym_file_dup, 'is objdump.basictags-i386 and basictags.symbols in sync');
  373. my $tags_obj_amd64 = Dpkg::Shlibs::Objdump::Object->new();
  374. open $objdump, '<', "$datadir/objdump.basictags-amd64"
  375. or die "$datadir/objdump.basictags-amd64: $!";
  376. $tags_obj_amd64->parse_objdump_output($objdump);
  377. close $objdump;
  378. # Merge/get_{new,lost} tests for optional tag:
  379. # - disappeared
  380. my $symbol11 = $tags_obj_i386->get_symbol('symbol11_optional@Base');
  381. delete $tags_obj_i386->{dynsyms}{'symbol11_optional@Base'};
  382. $sym_file->merge_symbols($tags_obj_i386, '100.MISSING');
  383. $sym = $sym_file->lookup_symbol('symbol11_optional@Base', ['libbasictags.so.1'], 1);
  384. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol11_optional@Base',
  385. symbol_templ => 'symbol11_optional@Base',
  386. minver => '1.1', dep_id => 1, deprecated => '100.MISSING',
  387. tags => { optional => undef }, tagorder => [ 'optional' ]),
  388. 'disappered optional symbol gets deprecated');
  389. $sym_file->merge_symbols($tags_obj_i386, '101.MISSING');
  390. $sym = $sym_file->lookup_symbol('symbol11_optional@Base', ['libbasictags.so.1'], 1);
  391. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol11_optional@Base',
  392. symbol_templ => 'symbol11_optional@Base',
  393. minver => '1.1', dep_id => 1, deprecated => '101.MISSING',
  394. tags => { optional => undef }, tagorder => [ 'optional' ]),
  395. 'deprecated text of MISSING optional symbol gets rebumped each merge');
  396. is( scalar($sym_file->get_lost_symbols($sym_file_dup)), 0, 'missing optional symbol is not LOST');
  397. # - reappeared (undeprecate, minver should be 1.1, not 100.MISSED)
  398. $tags_obj_i386->add_dynamic_symbol($symbol11);
  399. $sym_file->merge_symbols($tags_obj_i386, '100.MISSING');
  400. $sym = $sym_file->lookup_symbol('symbol11_optional@Base', ['libbasictags.so.1']);
  401. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol11_optional@Base',
  402. symbol_templ => 'symbol11_optional@Base',
  403. minver => '1.1', dep_id => 1, deprecated => 0,
  404. tags => { optional => undef }, tagorder => [ 'optional' ]),
  405. 'reappered optional symbol gets undeprecated + minver');
  406. is( scalar($sym_file->get_lost_symbols($sym_file_dup) +
  407. $sym_file->get_new_symbols($sym_file_dup)), 0, 'reappeared optional symbol: neither NEW nor LOST');
  408. # Merge/get_{new,lost} tests for arch tag:
  409. # - arch specific appears on wrong arch: 'arch' tag should be removed
  410. my $symbol21 = $tags_obj_amd64->get_symbol('symbol21_amd64@Base');
  411. $tags_obj_i386->add_dynamic_symbol($symbol21);
  412. $sym_file->merge_symbols($tags_obj_i386, '100.MISSING');
  413. $sym = $sym_file->lookup_symbol('symbol21_amd64@Base', ['libbasictags.so.1']);
  414. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol21_amd64@Base',
  415. symbol_templ => 'symbol21_amd64@Base', symbol_quoted => "'",
  416. minver => '2.1', dep_id => 0, deprecated => 0),
  417. 'symbol appears on foreign arch, arch tag should be removed');
  418. @tmp = map { $_->{symbol}->get_symbolname() } $sym_file->get_new_symbols($sym_file_dup);
  419. is_deeply( \@tmp, [ 'symbol21_amd64@Base' ], 'symbol from foreign arch is NEW');
  420. is( $sym->get_symbolspec(1), ' symbol21_amd64@Base 2.1', 'no tags => no quotes in the symbol name' );
  421. # - arch specific symbol disappears
  422. delete $tags_obj_i386->{dynsyms}{'symbol22_i386@Base'};
  423. delete $tags_obj_i386->{dynsyms}{'symbol24_32@Base'};
  424. delete $tags_obj_i386->{dynsyms}{'symbol26_little@Base'};
  425. delete $tags_obj_i386->{dynsyms}{'symbol28_little_32@Base'};
  426. delete $tags_obj_i386->{dynsyms}{'symbol41_i386_and_optional@Base'};
  427. $sym_file->merge_symbols($tags_obj_i386, '100.MISSING');
  428. $sym = $sym_file->lookup_symbol('symbol22_i386@Base', ['libbasictags.so.1'], 1);
  429. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol22_i386@Base',
  430. symbol_templ => 'symbol22_i386@Base',
  431. minver => '2.2', dep_id => 0, deprecated => '100.MISSING',
  432. tags => { arch => '!amd64 !ia64 !mips' },
  433. tagorder => [ 'arch' ]),
  434. 'disappeared arch specific symbol gets deprecated');
  435. $sym = $sym_file->lookup_symbol('symbol24_32@Base', ['libbasictags.so.1'], 1);
  436. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol24_32@Base',
  437. symbol_templ => 'symbol24_32@Base',
  438. minver => '2.4', dep_id => 0, deprecated => '100.MISSING',
  439. tags => { 'arch-bits' => '32' },
  440. tagorder => [ 'arch-bits' ]),
  441. 'disappeared arch bits specific symbol gets deprecated');
  442. $sym = $sym_file->lookup_symbol('symbol26_little@Base', ['libbasictags.so.1'], 1);
  443. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol26_little@Base',
  444. symbol_templ => 'symbol26_little@Base',
  445. minver => '2.6', dep_id => 0, deprecated => '100.MISSING',
  446. tags => { 'arch-endian' => 'little' },
  447. tagorder => [ 'arch-endian' ]),
  448. 'disappeared arch endian specific symbol gets deprecated');
  449. $sym = $sym_file->lookup_symbol('symbol28_little_32@Base', ['libbasictags.so.1'], 1);
  450. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol28_little_32@Base',
  451. symbol_templ => 'symbol28_little_32@Base',
  452. minver => '2.8', dep_id => 0, deprecated => '100.MISSING',
  453. tags => { 'arch-bits' => '32', 'arch-endian' => 'little' },
  454. tagorder => [ 'arch-bits', 'arch-endian' ]),
  455. 'disappeared arch bits and endian specific symbol gets deprecated');
  456. $sym = $sym_file->lookup_symbol('symbol41_i386_and_optional@Base', ['libbasictags.so.1'], 1);
  457. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol41_i386_and_optional@Base',
  458. symbol_templ => 'symbol41_i386_and_optional@Base',
  459. symbol_quoted => '"',
  460. minver => '4.1', dep_id => 0, deprecated => '100.MISSING',
  461. tags => { arch => 'i386', optional => 'reason' },
  462. tagorder => [ 'arch', 'optional' ]),
  463. 'disappeared optional arch specific symbol gets deprecated');
  464. @tmp = sort map { $_->{symbol}->get_symbolname() } $sym_file->get_lost_symbols($sym_file_dup);
  465. is_deeply(\@tmp, [ 'symbol22_i386@Base', 'symbol24_32@Base',
  466. 'symbol26_little@Base', 'symbol28_little_32@Base' ],
  467. "missing arch specific is LOST, but optional arch specific isn't");
  468. # Tests for tagged #includes
  469. $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/symbols.include-3", arch => 'i386');
  470. $sym = $sym_file->lookup_symbol('symbol2_fake1@Base', ['libbasictags.so.2']);
  471. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol2_fake1@Base',
  472. minver => '1.0',
  473. tags => { optional => undef, 'random tag' => 'random value' },
  474. tagorder => [ 'optional', 'random tag' ] ),
  475. 'symbols from #included file inherits tags');
  476. $sym = $sym_file->lookup_symbol('symbol41_i386_and_optional@Base', ['libbasictags.so.1']);
  477. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol41_i386_and_optional@Base',
  478. symbol_templ => 'symbol41_i386_and_optional@Base',
  479. symbol_quoted => '"',
  480. minver => '4.1',
  481. tags => { optional => 'reason', t => 'v', arch => 'i386' },
  482. tagorder => [ 'optional', 't', 'arch' ]),
  483. 'symbols in #included file can override tag values');
  484. $sym = $sym_file->lookup_symbol('symbol51_untagged@Base', ['libbasictags.so.1']);
  485. is_deeply($sym, Dpkg::Shlibs::Symbol->new(symbol => 'symbol51_untagged@Base',
  486. minver => '5.1',
  487. tags => { optional => 'from parent', t => 'v' },
  488. tagorder => [ 'optional', 't' ]),
  489. 'symbols are properly cloned when #including');
  490. # Test Symbol::clone()
  491. $sym = Dpkg::Shlibs::Symbol->new(symbol => 'foobar', testfield => 1, teststruct => { foo => 1 });
  492. $tmp = $sym->clone();
  493. $tmp->{teststruct}{foo} = 3;
  494. $tmp->{testfield} = 3;
  495. is ( $sym->{teststruct}{foo}, 1, 'original field "foo" not changed' );
  496. is ( $sym->{testfield}, 1, 'original field "testfield" not changed' );
  497. ############ Test symbol patterns ###########
  498. SKIP: {
  499. skip 'c++filt not available', 41 if not find_command('c++filt');
  500. sub load_patterns_obj {
  501. $obj = Dpkg::Shlibs::Objdump::Object->new();
  502. open $objdump, '<', "$datadir/objdump.patterns"
  503. or die "$datadir/objdump.patterns: $!";
  504. $obj->parse_objdump_output($objdump);
  505. close $objdump;
  506. return $obj;
  507. }
  508. sub load_patterns_symbols {
  509. $sym_file = Dpkg::Shlibs::SymbolFile->new(file => "$datadir/patterns.symbols");
  510. return $sym_file;
  511. }
  512. load_patterns_obj();
  513. $sym_file_dup = load_patterns_symbols();
  514. load_patterns_symbols();
  515. save_load_test($sym_file, 'save -> load test of patterns template', template_mode => 1);
  516. isnt( $sym_file->get_patterns('libpatterns.so.1') , 0,
  517. 'patterns.symbols has patterns');
  518. $sym_file->merge_symbols($obj, '100.MISSING');
  519. @tmp = map { $_->get_symbolname() } $sym_file->get_lost_symbols($sym_file_dup);
  520. is_deeply(\@tmp, [], 'no LOST symbols if all patterns matched.');
  521. @tmp = map { $_->get_symbolname() } $sym_file->get_new_symbols($sym_file_dup);
  522. is_deeply(\@tmp, [], 'no NEW symbols if all patterns matched.');
  523. # Pattern resolution order: aliases (c++, symver), generic
  524. $sym = $sym_file->lookup_symbol('SYMVER_1@SYMVER_1', 'libpatterns.so.1');
  525. is($sym->{minver}, '1', 'specific SYMVER_1 symbol');
  526. $sym = $sym_file->lookup_symbol('_ZN3NSB6Symver14symver_method1Ev@SYMVER_1', 'libpatterns.so.1');
  527. is($sym->{minver}, '1.method1', 'specific symbol preferred over pattern');
  528. $sym = $sym_file->lookup_symbol('_ZN3NSB6Symver14symver_method2Ev@SYMVER_1', 'libpatterns.so.1');
  529. is($sym->{minver}, '1.method2', 'c++ alias pattern preferred over generic pattern');
  530. is($sym->get_pattern()->get_symbolname(), 'NSB::Symver::symver_method2()@SYMVER_1',
  531. 'c++ alias pattern preferred over generic pattern, on demangled name');
  532. $sym = $sym_file->lookup_symbol('_ZN3NSB6SymverD1Ev@SYMVER_1', 'libpatterns.so.1');
  533. is ( $sym->{minver}, '1.generic', 'generic (c++ & symver) pattern covers the rest (destructor)' );
  534. ok($sym->get_pattern()->equals($sym_file->create_symbol('(c++|symver)SYMVER_1 1.generic')),
  535. 'generic (c++ & symver) pattern covers the rest (destructor), compared');
  536. # Test old style wildcard support
  537. load_patterns_symbols();
  538. $sym = $sym_file->create_symbol('*@SYMVEROPT_2 2');
  539. ok($sym->is_optional(), 'Old style wildcard is optional');
  540. is($sym->get_alias_type(), 'symver', 'old style wildcard is a symver pattern');
  541. is($sym->get_symbolname(), 'SYMVEROPT_2', 'wildcard pattern got renamed');
  542. $pat = $sym_file->lookup_pattern('(symver|optional)SYMVEROPT_2', 'libpatterns.so.1');
  543. $sym->{symbol_templ} = $pat->{symbol_templ};
  544. is_deeply($pat, $sym, 'old style wildcard is the same as (symver|optional)');
  545. # Get rid of all SymverOptional symbols
  546. foreach my $tmp (keys %{$obj->{dynsyms}}) {
  547. delete $obj->{dynsyms}{$tmp} if ($tmp =~ /SymverOptional/);
  548. }
  549. $sym_file->merge_symbols($obj, '100.MISSING');
  550. is_deeply ( [ map { $_->get_symbolname() } $pat->get_pattern_matches() ],
  551. [], 'old style wildcard matches nothing.');
  552. is($pat->{deprecated}, '100.MISSING', 'old style wildcard gets deprecated.');
  553. @tmp = map { $_->{symbol}->get_symbolname() } $sym_file->get_lost_symbols($sym_file_dup);
  554. is_deeply(\@tmp, [], 'but old style wildcard is not LOST.');
  555. # 'Internal' pattern covers all internal symbols
  556. load_patterns_obj();
  557. @tmp = grep { $_->get_symbolname() =~ /Internal/ } $sym_file->get_symbols('libpatterns.so.1');
  558. $sym = $sym_file->create_symbol('(regex|c++)^_Z(T[ISV])?N3NSA6ClassA8Internal.*@Base$ 1.internal');
  559. $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1');
  560. is_deeply ([ sort $pat->get_pattern_matches() ], [ sort @tmp ],
  561. 'Pattern covers all internal symbols');
  562. is($tmp[0]->{minver}, '1.internal', 'internal pattern covers first symbol');
  563. # Lookup private pattern
  564. my @private_symnames = sort qw(
  565. _ZN3NSA6ClassA7Private11privmethod1Ei@Base
  566. _ZN3NSA6ClassA7Private11privmethod2Ei@Base
  567. _ZN3NSA6ClassA7PrivateC1Ev@Base
  568. _ZN3NSA6ClassA7PrivateC2Ev@Base
  569. _ZN3NSA6ClassA7PrivateD0Ev@Base
  570. _ZN3NSA6ClassA7PrivateD1Ev@Base
  571. _ZN3NSA6ClassA7PrivateD2Ev@Base
  572. _ZTIN3NSA6ClassA7PrivateE@Base
  573. _ZTSN3NSA6ClassA7PrivateE@Base
  574. _ZTVN3NSA6ClassA7PrivateE@Base
  575. );
  576. $sym = $sym_file->create_symbol('(c++|regex|optional)NSA::ClassA::Private(::.*)?@Base 1');
  577. $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1');
  578. isnt( $pat, undef, 'pattern for private class has been found' );
  579. is_deeply( [ sort map { $_->get_symbolname() } $pat->get_pattern_matches() ],
  580. \@private_symnames, 'private pattern matched expected symbols');
  581. ok( ($pat->get_pattern_matches())[0]->is_optional(),
  582. 'private symbol is optional like its pattern');
  583. ok( $sym_file->lookup_symbol(($pat->get_pattern_matches())[0], 'libpatterns.so.1'),
  584. 'lookup_symbol() finds symbols matched by pattern (after merge)');
  585. # Get rid of a private symbol, it should not be lost
  586. delete $obj->{dynsyms}{$private_symnames[0]};
  587. load_patterns_symbols();
  588. $sym_file->merge_symbols($obj, '100.MISSING');
  589. $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1');
  590. @tmp = map { $_->{symbol}->get_symbolname() } $sym_file->get_lost_symbols($sym_file_dup);
  591. is_deeply(\@tmp, [], 'no LOST symbols when got rid of patterned optional symbol.');
  592. ok(!$pat->{deprecated}, 'there are still matches, pattern is not deprecated.');
  593. # Get rid of all private symbols, the pattern should be deprecated.
  594. foreach my $tmp (@private_symnames) {
  595. delete $obj->{dynsyms}{$tmp};
  596. }
  597. load_patterns_symbols();
  598. $sym_file->merge_symbols($obj, '100.MISSING');
  599. $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1', 1);
  600. @tmp = $sym_file->get_lost_symbols($sym_file_dup);
  601. is_deeply( \@tmp, [ ],
  602. 'All private symbols gone, but pattern is not LOST because it is optional.');
  603. is( $pat->{deprecated}, '100.MISSING',
  604. 'All private symbols gone - pattern deprecated.');
  605. # Internal symbols. All covered by the pattern?
  606. @tmp = grep { $_->get_symbolname() =~ /Internal/ } values %{$sym_file->{objects}{'libpatterns.so.1'}{syms}};
  607. $sym = $sym_file->create_symbol('(regex|c++)^_Z(T[ISV])?N3NSA6ClassA8Internal.*@Base$ 1.internal');
  608. $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1');
  609. is_deeply ([ sort $pat->get_pattern_matches() ], [ sort @tmp ],
  610. 'Pattern covers all internal symbols');
  611. is($tmp[0]->{minver}, '1.internal', 'internal pattern covers first symbol');
  612. # Delete matches of the non-optional pattern
  613. $sym = $sym_file->create_symbol('(c++)"non-virtual thunk to NSB::ClassD::generate_vt(char const*) const@Base" 1');
  614. $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1');
  615. isnt( $pat, undef, 'lookup_pattern() finds alias-based pattern' );
  616. is(scalar($pat->get_pattern_matches()), 2, 'two matches for the generate_vt pattern');
  617. foreach my $tmp ($pat->get_pattern_matches()) {
  618. delete $obj->{dynsyms}{$tmp->get_symbolname()};
  619. }
  620. load_patterns_symbols();
  621. $sym_file->merge_symbols($obj, '100.MISSING');
  622. $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1', 1);
  623. @tmp = map { scalar $sym_file->lookup_pattern($_->{symbol}, 'libpatterns.so.1', 1) }
  624. $sym_file->get_lost_symbols($sym_file_dup);
  625. is_deeply(\@tmp, [ $pat ], 'No matches - generate_vt() pattern is LOST.');
  626. is( $pat->{deprecated}, '100.MISSING',
  627. 'No matches - generate_vt() pattern is deprecated.');
  628. # Pattern undeprecation when matches are discovered
  629. load_patterns_obj();
  630. load_patterns_symbols();
  631. $pat = $sym_file_dup->lookup_pattern($sym, 'libpatterns.so.1');
  632. $pat->{deprecated} = '0.1-1';
  633. $pat = $sym_file->lookup_pattern($sym, 'libpatterns.so.1');
  634. $pat->{deprecated} = '0.1-1';
  635. $sym_file->merge_symbols($obj, '100.FOUND');
  636. ok( ! $pat->{deprecated},
  637. 'Previously deprecated pattern with matches got undeprecated');
  638. is( $pat->{minver}, '100.FOUND',
  639. 'Previously deprecated pattern with matches got minver bumped');
  640. @tmp = map { $_->{symbol}->get_symbolspec(1) } $sym_file->get_new_symbols($sym_file_dup);
  641. is_deeply( \@tmp, [ $pat->get_symbolspec(1) ],
  642. 'Previously deprecated pattern with matches is NEW. Matches themselves are not NEW.');
  643. foreach my $sym ($pat->get_pattern_matches()) {
  644. ok(!$sym->{deprecated}, $sym->get_symbolname() . ': not deprecated');
  645. is($sym->{minver}, '100.FOUND', $sym->get_symbolname() . ': version bumped');
  646. }
  647. }