3 // Copyright (C) 2004, 2005, 2006, 2007, 2008 Free Software Foundation, Inc.
5 // This library is free software; you can redistribute it and/or
6 // modify it under the terms of the GNU General Public License as
7 // published by the Free Software Foundation; either version 2, or (at
8 // your option) any later version.
10 // This library is distributed in the hope that it will be useful, but
11 // WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 // General Public License for more details.
15 // You should have received a copy of the GNU General Public License
16 // along with this library; see the file COPYING. If not, write to
17 // the Free Software Foundation, 51 Franklin Street, Fifth Floor, Boston,
18 // MA 02110-1301, USA.
20 // As a special exception, you may use this file as part of a free
21 // software library without restriction. Specifically, if other files
22 // instantiate templates or use macros or inline functions from this
23 // file, or you compile this file and link it with other files to
24 // produce an executable, this file does not by itself cause the
25 // resulting executable to be covered by the GNU General Public
26 // License. This exception does not however invalidate any other
27 // reasons why the executable file might be covered by the GNU General
30 // Benjamin Kosnik <bkoz@redhat.com>
32 #include "testsuite_abi.h"
43 symbol::init(string
& data
)
45 const char delim
= ':';
46 const char version_delim
= '@';
47 const string::size_type npos
= string::npos
;
48 string::size_type n
= 0;
51 if (data
.find("FUNC") == 0)
52 type
= symbol::function
;
53 else if (data
.find("OBJECT") == 0)
54 type
= symbol::object
;
56 n
= data
.find_first_of(delim
);
58 data
.erase(data
.begin(), data
.begin() + n
+ 1);
60 // Iff object, get size info.
61 if (type
== symbol::object
)
63 n
= data
.find_first_of(delim
);
66 string
objectsize(data
.begin(), data
.begin() + n
);
67 istringstream
iss(objectsize
);
72 data
.erase(data
.begin(), data
.begin() + n
+ 1);
76 // Set the name and raw_name.
77 raw_name
= string(data
.begin(), data
.end());
78 n
= data
.find_first_of(version_delim
);
81 // Found version string.
82 name
= string(data
.begin(), data
.begin() + n
);
83 n
= data
.find_last_of(version_delim
);
84 data
.erase(data
.begin(), data
.begin() + n
+ 1);
91 // No versioning info.
92 name
= string(data
.begin(), data
.end());
93 version_status
= symbol::none
;
96 // Set the demangled name.
97 demangled_name
= demangle(name
);
101 symbol::print() const
103 const char tab
= '\t';
104 cout
<< name
<< endl
;
106 if (demangled_name
!= name
)
107 cout
<< demangled_name
<< endl
;
110 switch (version_status
)
119 vers
= "incompatible";
122 vers
= "unversioned";
127 cout
<< "version status: " << vers
<< endl
;
129 if (version_name
.size()
130 && (version_status
== compatible
|| version_status
== incompatible
))
131 cout
<< version_name
<< endl
;
137 type_string
= "function";
140 type_string
= "object";
143 type_string
= "uncategorized";
146 type_string
= "<default>";
148 cout
<< "type: " << type_string
<< endl
;
151 cout
<< "type size: " << size
<< endl
;
153 string status_string
;
157 status_string
= "added";
160 status_string
= "subtracted";
163 status_string
= "undesignated";
166 status_string
= "<default>";
168 cout
<< "status: " << status_string
<< endl
;
175 check_version(symbol
& test
, bool added
)
177 // Construct list of compatible versions.
178 typedef std::vector
<std::string
> compat_list
;
179 static compat_list known_versions
;
180 if (known_versions
.empty())
182 // NB: First version here must be the default version for this
183 // version of DT_SONAME.
184 known_versions
.push_back("GLIBCXX_3.4");
185 known_versions
.push_back("GLIBCXX_3.4.1");
186 known_versions
.push_back("GLIBCXX_3.4.2");
187 known_versions
.push_back("GLIBCXX_3.4.3");
188 known_versions
.push_back("GLIBCXX_3.4.4");
189 known_versions
.push_back("GLIBCXX_3.4.5");
190 known_versions
.push_back("GLIBCXX_3.4.6");
191 known_versions
.push_back("GLIBCXX_3.4.7");
192 known_versions
.push_back("GLIBCXX_3.4.8");
193 known_versions
.push_back("GLIBCXX_3.4.9");
194 known_versions
.push_back("GLIBCXX_3.4.10");
195 known_versions
.push_back("GLIBCXX_3.4.11");
196 known_versions
.push_back("GLIBCXX_LDBL_3.4");
197 known_versions
.push_back("GLIBCXX_LDBL_3.4.7");
198 known_versions
.push_back("GLIBCXX_LDBL_3.4.10");
199 known_versions
.push_back("CXXABI_1.3");
200 known_versions
.push_back("CXXABI_1.3.1");
201 known_versions
.push_back("CXXABI_1.3.2");
202 known_versions
.push_back("CXXABI_1.3.3");
203 known_versions
.push_back("CXXABI_LDBL_1.3");
205 compat_list::iterator begin
= known_versions
.begin();
206 compat_list::iterator end
= known_versions
.end();
208 // Check for compatible version.
209 if (test
.version_name
.size())
211 compat_list::iterator it1
= find(begin
, end
, test
.version_name
);
212 compat_list::iterator it2
= find(begin
, end
, test
.name
);
214 test
.version_status
= symbol::compatible
;
216 test
.version_status
= symbol::incompatible
;
218 // Check that added symbols aren't added in the base version.
219 if (added
&& test
.version_name
== known_versions
[0])
220 test
.version_status
= symbol::incompatible
;
222 // Check that long double compatibility symbols demangled as
223 // __float128 are put into some _LDBL_ version name.
224 if (added
&& test
.demangled_name
.find("__float128") != std::string::npos
)
226 // Has to be in _LDBL_ version name.
227 if (test
.version_name
.find("_LDBL_") == std::string::npos
)
228 test
.version_status
= symbol::incompatible
;
231 // Check for weak label.
232 if (it1
== end
&& it2
== end
)
233 test
.version_status
= symbol::incompatible
;
238 // version as compatible
245 // New version labels are ok. The rest are not.
246 compat_list::iterator it2
= find(begin
, end
, test
.name
);
248 test
.version_status
= symbol::compatible
;
250 test
.version_status
= symbol::incompatible
;
253 return test
.version_status
== symbol::compatible
;
257 check_compatible(symbol
& lhs
, symbol
& rhs
, bool verbose
)
260 const char tab
= '\t';
262 // Check to see if symbol_objects are compatible.
263 if (lhs
.type
!= rhs
.type
)
267 cout
<< tab
<< "incompatible types" << endl
;
270 if (lhs
.name
!= rhs
.name
)
274 cout
<< tab
<< "incompatible names" << endl
;
277 if (lhs
.size
!= rhs
.size
)
282 cout
<< tab
<< "incompatible sizes" << endl
;
283 cout
<< tab
<< lhs
.size
<< endl
;
284 cout
<< tab
<< rhs
.size
<< endl
;
288 if (lhs
.version_name
!= rhs
.version_name
289 && !check_version(lhs
) && !check_version(rhs
))
294 cout
<< tab
<< "incompatible versions" << endl
;
295 cout
<< tab
<< lhs
.version_name
<< endl
;
296 cout
<< tab
<< rhs
.version_name
<< endl
;
308 has_symbol(const string
& name
, const symbols
& s
) throw()
309 { return s
.find(name
) != s
.end(); }
312 get_symbol(const string
& name
, const symbols
& s
)
314 symbols::const_iterator i
= s
.find(name
);
322 os
<< "get_symbol failed for symbol " << name
;
323 __throw_logic_error(os
.str().c_str());
328 examine_symbol(const char* name
, const char* file
)
332 symbols s
= create_symbols(file
);
333 const symbol
& sym
= get_symbol(name
, s
);
337 { __throw_exception_again
; }
341 compare_symbols(const char* baseline_file
, const char* test_file
,
344 // Input both lists of symbols into container.
345 symbols baseline
= create_symbols(baseline_file
);
346 symbols test
= create_symbols(test_file
);
348 // Sanity check results.
349 if (!baseline
.size() || !test
.size())
351 cerr
<< "Problems parsing the list of exported symbols." << endl
;
355 // Check to see if any long double compatibility symbols are produced.
356 bool ld_version_found(false);
357 symbols::iterator
li(test
.begin());
358 while (!ld_version_found
&& li
!= test
.end())
360 if (li
->second
.version_name
.find("_LDBL_") != std::string::npos
)
361 ld_version_found
= true;
366 // Assuming all baseline names and test names are both unique w/ no
369 // The names added to missing_names are baseline names not found in
371 // -> symbols that have been deleted.
373 // The names added to added_names are test names not in
375 // -> symbols that have been added.
376 typedef std::vector
<std::string
> symbol_names
;
377 symbol_names shared_names
;
378 symbol_names missing_names
;
379 symbol_names added_names
;
380 for (li
= test
.begin(); li
!= test
.end(); ++li
)
381 added_names
.push_back(li
->first
);
383 for (symbols::iterator i
= baseline
.begin(); i
!= baseline
.end(); ++i
)
385 string
name(i
->first
);
386 symbol_names::iterator end
= added_names
.end();
387 symbol_names::iterator it
= find(added_names
.begin(), end
, name
);
391 shared_names
.push_back(name
);
392 added_names
.erase(it
);
396 // Iff no test long double compatibility symbols at all and the symbol
397 // missing is a baseline long double compatibility symbol, skip.
398 string
version_name(i
->second
.version_name
);
399 bool base_ld(version_name
.find("_LDBL_") != std::string::npos
);
400 if (!base_ld
|| base_ld
&& ld_version_found
)
401 missing_names
.push_back(name
);
405 // Fill out list of incompatible symbols.
406 typedef pair
<symbol
, symbol
> symbol_pair
;
407 vector
<symbol_pair
> incompatible
;
409 // Check missing names for compatibility.
410 for (size_t j
= 0; j
< missing_names
.size(); ++j
)
412 symbol
& sbase
= baseline
[missing_names
[j
]];
413 sbase
.status
= symbol::subtracted
;
414 incompatible
.push_back(symbol_pair(sbase
, sbase
));
417 // Check shared names for compatibility.
418 const symbol_names::size_type shared_size
= shared_names
.size();
419 for (size_t k
= 0; k
< shared_size
; ++k
)
421 symbol
& sbase
= baseline
[shared_names
[k
]];
422 symbol
& stest
= test
[shared_names
[k
]];
423 stest
.status
= symbol::existing
;
424 if (!check_compatible(sbase
, stest
))
425 incompatible
.push_back(symbol_pair(sbase
, stest
));
428 // Check added names for compatibility.
429 const symbol_names::size_type added_size
= added_names
.size();
430 for (size_t l
= 0; l
< added_size
; ++l
)
432 symbol
& stest
= test
[added_names
[l
]];
433 stest
.status
= symbol::added
;
434 if (!check_version(stest
, true))
435 incompatible
.push_back(symbol_pair(stest
, stest
));
439 if (verbose
&& added_names
.size())
441 cout
<< endl
<< added_names
.size() << " added symbols " << endl
;
442 for (size_t j
= 0; j
< added_names
.size() ; ++j
)
445 test
[added_names
[j
]].print();
449 if (verbose
&& missing_names
.size())
451 cout
<< endl
<< missing_names
.size() << " missing symbols " << endl
;
452 for (size_t j
= 0; j
< missing_names
.size() ; ++j
)
455 baseline
[missing_names
[j
]].print();
459 if (verbose
&& incompatible
.size())
461 cout
<< endl
<< incompatible
.size() << " incompatible symbols " << endl
;
462 for (size_t j
= 0; j
< incompatible
.size() ; ++j
)
464 // First, print index.
467 // Second, report name.
468 symbol
& sbase
= incompatible
[j
].first
;
469 symbol
& stest
= incompatible
[j
].second
;
472 // Second, report reason or reasons incompatible.
473 check_compatible(sbase
, stest
, true);
477 cout
<< "\n\t\t=== libstdc++-v3 check-abi Summary ===" << endl
;
479 cout
<< "# of added symbols:\t\t " << added_names
.size() << endl
;
480 cout
<< "# of missing symbols:\t\t " << missing_names
.size() << endl
;
481 cout
<< "# of incompatible symbols:\t " << incompatible
.size() << endl
;
483 cout
<< "using: " << baseline_file
<< endl
;
485 return !(missing_names
.size() || incompatible
.size());
490 create_symbols(const char* file
)
496 // Organize file data into an associated container (symbols) of symbol
497 // objects mapped to mangled names without versioning
501 while (getline(ifs
, line
).good())
512 os
<< "create_symbols failed for file " << file
;
513 __throw_runtime_error(os
.str().c_str());
520 demangle(const std::string
& mangled
)
523 if (mangled
[0] != '_' || mangled
[1] != 'Z')
525 // This is not a mangled symbol, thus has "C" linkage.
526 name
= mangled
.c_str();
530 // Use __cxa_demangle to demangle.
532 name
= abi::__cxa_demangle(mangled
.c_str(), 0, 0, &status
);
538 name
= "error code = 0: success";
541 name
= "error code = -1: memory allocation failure";
544 name
= "error code = -2: invalid mangled name";
547 name
= "error code = -3: invalid arguments";
550 name
= "error code unknown - who knows what happened";