20#include "abg-reporter-priv.h"
115 for (string_enumerator_map::const_iterator i = enumerators_map.begin();
116 i != enumerators_map.end();
118 sorted.push_back(i->second);
120 std::sort(sorted.begin(), sorted.end(), comp);
132 for (string_changed_enumerator_map::const_iterator i =
133 enumerators_map.begin();
134 i != enumerators_map.end();
136 sorted.push_back(i->second);
139 std::sort(sorted.begin(), sorted.end(), comp);
149 vector<decl_base_sptr>& sorted)
151 sorted.reserve(data_members.size());
152 for (string_decl_base_sptr_map::const_iterator i = data_members.begin();
153 i != data_members.end();
155 sorted.push_back(i->second);
158 std::sort(sorted.begin(), sorted.end(), comp);
168 std::sort(to_sort.begin(), to_sort.end(), comp);
181get_symbol(
const decl_base_sptr& d)
184 return fn->get_symbol();
186 return var->get_symbol();
205 string fr = f->get_qualified_name(), sr = s->get_qualified_name();
210 if (!f->get_linkage_name().empty()
211 && !s->get_linkage_name().empty())
213 fr = f->get_linkage_name();
214 sr = s->get_linkage_name();
219 if (get_symbol(f) && get_symbol(s))
221 fr = get_symbol(f)->get_id_string();
222 sr = get_symbol(s)->get_id_string();
227 fr = f->get_pretty_representation(
true,
true);
228 sr = s->get_pretty_representation(
true,
true);
245 if (!first || !second)
259 vector<const function_decl*>& sorted)
261 sorted.reserve(map.size());
262 for (string_function_ptr_map::const_iterator i = map.begin();
265 sorted.push_back(i->second);
268 std::sort(sorted.begin(), sorted.end(), comp);
282 sorted.reserve(map.size());
283 for (string_member_function_sptr_map::const_iterator i = map.begin();
286 sorted.push_back(i->second);
289 std::sort(sorted.begin(), sorted.end(), comp);
305 sorted.reserve(map.size());
306 for (string_function_decl_diff_sptr_map::const_iterator i = map.begin();
309 sorted.push_back(i->second);
311 std::sort(sorted.begin(), sorted.end(), comp);
324 std::sort(fn_diffs.begin(), fn_diffs.end(), comp);
337 sorted.reserve(map.size());
338 for (string_var_diff_sptr_map::const_iterator i = map.begin();
341 sorted.push_back(i->second);
344 std::sort(sorted.begin(), sorted.end(), comp);
357 std::sort(var_diffs.begin(), var_diffs.end(), comp);
371 vector<elf_symbol_sptr>& sorted)
373 for (string_elf_symbol_map::const_iterator i = map.begin();
376 sorted.push_back(i->second);
379 std::sort(sorted.begin(), sorted.end(), comp);
392 vector<var_decl_sptr>& sorted)
394 for (string_var_ptr_map::const_iterator i = map.begin();
397 sorted.push_back(i->second);
400 std::sort(sorted.begin(), sorted.end(), comp);
413 sorted.reserve(map.size());
414 for (string_var_diff_sptr_map::const_iterator i = map.begin();
417 sorted.push_back(i->second);
419 std::sort(sorted.begin(), sorted.end(), comp);
432 sorted.reserve(map.size());
433 for (unsigned_var_diff_sptr_map::const_iterator i = map.begin();
436 sorted.push_back(i->second);
438 std::sort(sorted.begin(), sorted.end(), comp);
454 sorted.reserve(map.size());
455 for (string_function_decl_diff_sptr_map::const_iterator i = map.begin();
458 sorted.push_back(i->second);
461 sort(sorted.begin(), sorted.end(), comp);
475 sorted.reserve(map.size());
476 for (string_diff_sptr_map::const_iterator i = map.begin();
479 sorted.push_back(i->second);
482 sort(sorted.begin(), sorted.end(), comp);
496 sorted.reserve(map.size());
497 for (string_diff_ptr_map::const_iterator i = map.begin();
500 sorted.push_back(i->second);
503 sort(sorted.begin(), sorted.end(), comp);
517 for (string_base_diff_sptr_map::const_iterator i = map.begin();
520 sorted.push_back(i->second);
522 sort(sorted.begin(), sorted.end(), comp);
531 for (string_base_sptr_map::const_iterator i = m.begin();
534 sorted.push_back(i->second);
537 std::sort(sorted.begin(), sorted.end(), comp);
549 vector<fn_parm_diff_sptr>& sorted)
551 sorted.reserve(map.size());
552 for (unsigned_fn_parm_diff_sptr_map::const_iterator i = map.begin();
555 sorted.push_back(i->second);
558 std::sort(sorted.begin(), sorted.end(), comp);
570 vector<fn_parm_diff_sptr>& sorted)
572 sorted.reserve(map.size());
573 for (string_fn_parm_diff_sptr_map::const_iterator i = map.begin();
576 sorted.push_back(i->second);
579 std::sort(sorted.begin(), sorted.end(), comp);
590 vector<function_decl::parameter_sptr>& sorted)
592 for (string_parm_map::const_iterator i = map.begin();
595 sorted.push_back(i->second);
598 std::sort(sorted.begin(), sorted.end(), comp);
610 vector<type_or_decl_base_sptr>& sorted)
613 for (artifact_sptr_set_type::const_iterator it = set.begin();
616 sorted.push_back(*it);
619 std::sort(sorted.begin(), sorted.end(), comp);
634 vector<type_base_sptr>& sorted)
636 for (string_type_base_sptr_map::const_iterator i = map.begin();
639 sorted.push_back(i->second);
642 std::sort(sorted.begin(), sorted.end(), comp);
654 return type_base_sptr();
656 type_base_sptr ut = t->get_underlying_type();
657 qualified_type_def_sptr qut = dynamic_pointer_cast<qualified_type_def>(ut);
685 if (decl_base_sptr decl =
is_decl(first))
702 |
static_cast<unsigned>(r));}
709 &
static_cast<unsigned>(r));
715{
return static_cast<visiting_kind>(~static_cast<unsigned>(l));}
788 if (dif->first_class_or_union()->get_is_anonymous())
824 if (dif->first_subrange()->get_is_anonymous())
861 if (d->has_local_changes())
1035diff_context::diff_context()
1051diff_context::~diff_context() =
default;
1058{
return priv_->do_log_;}
1065{priv_->do_log_ = f;}
1072{priv_->corpus_diff_ = d;}
1079{
return priv_->corpus_diff_;}
1088 if (priv_->corpus_diff_)
1089 return priv_->corpus_diff_->first_corpus();
1090 return corpus_sptr();
1101 if (priv_->corpus_diff_)
1102 return priv_->corpus_diff_->second_corpus();
1103 return corpus_sptr();
1112 if (!priv_->reporter_)
1120 return priv_->reporter_;
1128{priv_->reporter_ = r;}
1142 types_or_decls_diff_map_type::const_iterator i =
1143 priv_->types_or_decls_diff_map.find(std::make_pair(first, second));
1144 if (i != priv_->types_or_decls_diff_map.end())
1158diff_context::has_diff_for_types(
const type_base_sptr first,
1159 const type_base_sptr second)
const
1160{
return has_diff_for(first, second);}
1168diff_context::has_diff_for(
const diff* d)
const
1169{
return has_diff_for(d->first_subject(), d->second_subject()).get();}
1177diff_context::has_diff_for(
const diff_sptr d)
const
1178{
return has_diff_for(d->first_subject(), d->second_subject());}
1187{
return priv_->allowed_category_;}
1196{priv_->allowed_category_ = c;}
1209{priv_->allowed_category_ = priv_->allowed_category_ | c;}
1220{priv_->allowed_category_ = priv_->allowed_category_ & ~c;}
1236{priv_->types_or_decls_diff_map[std::make_pair(first, second)] = d;}
1242diff_context::add_diff(
const diff* d)
1255diff_context::add_diff(
const diff_sptr d)
1258 add_diff(d->first_subject(), d->second_subject(), d);
1275{
return has_diff_for(first, second);}
1287{
return has_diff_for(d);}
1303 if (!has_diff_for(first, second))
1305 add_diff(first, second, d);
1306 priv_->canonical_diffs.push_back(d);
1331 canonical = canonical_diff;
1332 set_canonical_diff_for(first, second, canonical);
1375{priv_->live_diffs_.insert(d);}
1388 size_t ptr_value =
reinterpret_cast<size_t>(canonical);
1389 pointer_map::iterator it = priv_->visited_diff_nodes_.find(ptr_value);
1390 if (it != priv_->visited_diff_nodes_.end())
1391 return reinterpret_cast<diff*
>(it->second);
1424 size_t canonical_ptr_value =
reinterpret_cast<size_t>(canonical);
1425 size_t diff_ptr_value =
reinterpret_cast<size_t>(d);
1426 priv_->visited_diff_nodes_[canonical_ptr_value] = diff_ptr_value;
1432{priv_->visited_diff_nodes_.clear();}
1442{priv_->forbid_visiting_a_node_twice_ = f;}
1452{priv_->reset_visited_diffs_for_each_interface_ = f;}
1460{
return priv_->forbid_visiting_a_node_twice_;}
1474 return (priv_->forbid_visiting_a_node_twice_
1475 && priv_->reset_visited_diffs_for_each_interface_);
1483{
return priv_->filters_;}
1491{priv_->filters_.push_back(f);}
1509 for (filtering::filters::const_iterator i =
diff_filters().begin();
1516 std::cerr <<
"applying a filter to diff '"
1527 std::cerr <<
"filter applied!:" << t <<
"\n";
1529 std::cerr <<
"propagating categories for the same diff node ... \n";
1538 std::cerr <<
"category propagated!: " << t <<
"\n";
1559 for (filtering::filters::const_iterator i =
diff_filters().begin();
1574{
return priv_->suppressions_;}
1585 priv_->negated_suppressions_.clear();
1586 priv_->direct_suppressions_.clear();
1587 return priv_->suppressions_;
1606 if (priv_->negated_suppressions_.empty())
1609 priv_->negated_suppressions_.push_back(s);
1611 return priv_->negated_suppressions_;
1629 if (priv_->direct_suppressions_.empty())
1633 priv_->direct_suppressions_.push_back(s);
1635 return priv_->direct_suppressions_;
1646 priv_->suppressions_.push_back(suppr);
1649 priv_->negated_suppressions_.clear();
1650 priv_->direct_suppressions_.clear();
1661 priv_->suppressions_.insert(priv_->suppressions_.end(),
1662 supprs.begin(), supprs.end());
1671{
return priv_->perform_change_categorization_;}
1678{priv_->perform_change_categorization_ = f;}
1692 priv_->leaf_changes_only_ = f;
1702{
return priv_->leaf_changes_only_;}
1712{
return priv_->hex_values_;}
1722{priv_->hex_values_ = f;}
1731{
return priv_->show_offsets_sizes_in_bits_;}
1740{priv_->show_offsets_sizes_in_bits_ = f;}
1749{priv_->show_relative_offset_changes_ = f;}
1758{
return priv_->show_relative_offset_changes_;}
1766{priv_->show_stats_only_ = f;}
1774{
return priv_->show_stats_only_;}
1782{priv_->show_soname_change_ = f;}
1790{
return priv_->show_soname_change_;}
1798{priv_->show_architecture_change_ = f;}
1806{
return priv_->show_architecture_change_;}
1813{priv_->show_deleted_fns_ = f;}
1819{
return priv_->show_deleted_fns_;}
1826{priv_->show_changed_fns_ = f;}
1831{
return priv_->show_changed_fns_;}
1838{priv_->show_added_fns_ = f;}
1844{
return priv_->show_added_fns_;}
1851{priv_->show_deleted_vars_ = f;}
1857{
return priv_->show_deleted_vars_;}
1864{priv_->show_changed_vars_ = f;}
1869{
return priv_->show_changed_vars_;}
1876{priv_->show_added_vars_ = f;}
1882{
return priv_->show_added_vars_;}
1885diff_context::show_linkage_names()
const
1886{
return priv_->show_linkage_names_;}
1889diff_context::show_linkage_names(
bool f)
1890{priv_->show_linkage_names_= f;}
1897{priv_->show_locs_= f;}
1903{
return priv_->show_locs_;}
1912{
return priv_->show_redundant_changes_;}
1921{priv_->show_redundant_changes_ = f;}
1929{
return priv_->show_syms_unreferenced_by_di_;}
1937{priv_->show_syms_unreferenced_by_di_ = f;}
1946{
return priv_->show_added_syms_unreferenced_by_di_;}
1955{priv_->show_added_syms_unreferenced_by_di_ = f;}
1964{priv_->show_unreachable_types_ = f;}
1973{
return priv_->show_unreachable_types_;}
1984{
return priv_->show_impacted_interfaces_;}
1995{priv_->show_impacted_interfaces_ = f;}
2004{priv_->default_output_stream_ = o;}
2013{
return priv_->default_output_stream_;}
2021{priv_->error_output_stream_ = o;}
2029{
return priv_->error_output_stream_;}
2038{
return priv_->dump_diff_tree_;}
2047{priv_->dump_diff_tree_ = f;}
2086 : priv_(new priv(first_subject, second_subject,
2108 : priv_(new priv(first_subject, second_subject,
2147 if (priv_->canonical_diff_)
2148 priv_->canonical_diff_->priv_->traversing_ =
true;
2149 priv_->traversing_ =
true;
2166 if (priv_->canonical_diff_)
2167 return priv_->canonical_diff_->priv_->traversing_;
2168 return priv_->traversing_;
2182 if (priv_->canonical_diff_)
2183 priv_->canonical_diff_->priv_->traversing_ =
false;
2184 priv_->traversing_ =
false;
2200 if (diff::priv_->finished_)
2203 diff::priv_->finished_ =
true;
2211{
return dynamic_pointer_cast<type_or_decl_base>(priv_->first_subject_);}
2218{
return dynamic_pointer_cast<type_or_decl_base>(priv_->second_subject_);}
2225{
return priv_->children_;}
2232{
return priv_->parent_;}
2245{
return priv_->canonical_diff_;}
2253{priv_->canonical_diff_ = d;}
2266 context()->keep_diff_alive(d);
2273 priv_->children_.push_back(d.get());
2275 d->priv_->parent_ =
this;
2283{
return priv_->get_context();}
2300 if (priv_->canonical_diff_)
2301 return priv_->canonical_diff_->priv_->currently_reporting_;
2302 return priv_->currently_reporting_;
2313 if (priv_->canonical_diff_)
2314 priv_->canonical_diff_->priv_->currently_reporting_ = f;
2315 priv_->currently_reporting_ = f;
2326 return priv_->canonical_diff_->priv_->reported_once_;
2386 bool already_visited =
false;
2387 if (
context()->visiting_a_node_twice_is_forbidden()
2388 &&
context()->diff_has_been_visited(
this))
2389 already_visited =
true;
2391 bool mark_visited_nodes_as_traversed =
2394 if (!already_visited && !v.
visit(
this,
true))
2397 if (mark_visited_nodes_as_traversed)
2398 context()->mark_diff_as_visited(
this);
2404 && !already_visited)
2411 if (!(*i)->traverse(v))
2414 if (mark_visited_nodes_as_traversed)
2415 context()->mark_diff_as_visited(
this);
2423 if (!v.
visit(
this,
false))
2426 if (mark_visited_nodes_as_traversed)
2427 context()->mark_diff_as_visited(
this);
2432 if (!already_visited && mark_visited_nodes_as_traversed)
2433 context()->mark_diff_as_visited(
this);
2447 priv_->canonical_diff_->priv_->reported_once_ = f;
2448 priv_->reported_once_ = f;
2460{
return priv_->local_category_;}
2486{
return priv_->category_;}
2501 priv_->category_ = priv_->category_ | c;
2502 return priv_->category_;
2516 priv_->local_category_ = priv_->local_category_ | c;
2517 return priv_->local_category_;
2545 priv_->category_ = priv_->category_ & ~c;
2546 return priv_->category_;
2560 priv_->local_category_ = priv_->local_category_ & ~c;
2561 return priv_->local_category_;
2571{priv_->category_ = c;}
2578{priv_->local_category_ = c;}
2603 && !canonical->is_allowed_by_specific_negated_suppression()
2604 && !canonical->has_descendant_allowed_by_specific_negated_suppression()
2605 && !canonical->has_parent_allowed_by_specific_negated_suppression())
2646 return priv_->is_filtered_out(c);
2657 bool is_private =
false;
2686 bool do_suppress = !
context()->negated_suppressions().empty();
2690 for (
auto n :
context()->negated_suppressions())
2691 if (!n->suppresses_diff(
this))
2693 do_suppress =
false;
2700 for (
auto d :
context()->direct_suppressions())
2701 if (d->suppresses_diff(
this))
2705 is_private_type =
true;
2759 for (suppressions_type::const_iterator i = suppressions.begin();
2760 i != suppressions.end();
2764 && !(*i)->suppresses_diff(
this))
2804 if (priv_->pretty_representation_.empty())
2805 priv_->pretty_representation_ =
"empty_diff";
2806 return priv_->pretty_representation_;
2833type_diff_base::type_diff_base(type_base_sptr first_subject,
2834 type_base_sptr second_subject,
2836 :
diff(first_subject, second_subject, ctxt),
2840type_diff_base::~type_diff_base()
2856 decl_base_sptr second_subject,
2858 :
diff(first_subject, second_subject, ctxt),
2862decl_diff_base::~decl_diff_base()
2873 if (diff::priv_->pretty_representation_.empty())
2875 std::ostringstream o;
2876 o <<
"distinct_diff[";
2887 diff::priv_->pretty_representation_ = o.str();
2889 return diff::priv_->pretty_representation_;
2922 :
diff(first, second, ctxt),
2956 if (!priv_->compatible_child_diff)
2964 priv_->compatible_child_diff =
2970 return priv_->compatible_child_diff;
2995 return typeid(f) !=
typeid(s);
3013 return NO_CHANGE_KIND;
3024 context()->get_reporter()->report(*
this, out, indent);
3047 ctxt->initialize_canonical_diff(result);
3072template<
typename DiffType>
3078 if (shared_ptr<DiffType> f =
3079 dynamic_pointer_cast<DiffType>(first))
3081 shared_ptr<DiffType> s =
3082 dynamic_pointer_cast<DiffType>(second);
3108 dynamic_pointer_cast<class_decl>(first))
3114 if (f->get_is_declaration_only())
3121 if (s->get_is_declaration_only())
3176 ((d = try_to_diff<type_decl>(f, s, ctxt))
3177 ||(d = try_to_diff<enum_type_decl>(f, s, ctxt))
3178 ||(d = try_to_diff<union_decl>(f, s,ctxt))
3180 ||(d = try_to_diff<pointer_type_def>(f, s, ctxt))
3181 ||(d = try_to_diff<reference_type_def>(f, s, ctxt))
3182 ||(d = try_to_diff<ptr_to_mbr_type>(f, s, ctxt))
3183 ||(d = try_to_diff<array_type_def::subrange_type>(f, s, ctxt))
3184 ||(d = try_to_diff<array_type_def>(f, s, ctxt))
3185 ||(d = try_to_diff<qualified_type_def>(f, s, ctxt))
3186 ||(d = try_to_diff<typedef_decl>(f, s, ctxt))
3187 ||(d = try_to_diff<function_type>(f, s, ctxt))
3188 ||(d = try_to_diff_distinct_kinds(f, s, ctxt)));
3198 |
static_cast<unsigned>(c2));}
3217 ^
static_cast<unsigned>(c2));}
3222 &
static_cast<unsigned>(c2));}
3226{
return static_cast<diff_category>(~static_cast<unsigned>(c));}
3265 | abigail::comparison::REFERENCE_LVALUENESS_CHANGE_CATEGORY
3307 bool emitted_a_category =
false;
3311 o <<
"NO_CHANGE_CATEGORY";
3312 emitted_a_category =
true;
3317 if (emitted_a_category)
3319 o <<
"ACCESS_CHANGE_CATEGORY";
3320 emitted_a_category |=
true;
3325 if (emitted_a_category)
3327 o <<
"COMPATIBLE_TYPE_CHANGE_CATEGORY";
3328 emitted_a_category |=
true;
3333 if (emitted_a_category)
3335 o <<
"HARMLESS_DECL_NAME_CHANGE_CATEGORY";
3336 emitted_a_category |=
true;
3341 if (emitted_a_category)
3343 o <<
"NON_VIRT_MEM_FUN_CHANGE_CATEGORY";
3344 emitted_a_category |=
true;
3349 if (emitted_a_category)
3351 o <<
"STATIC_DATA_MEMBER_CHANGE_CATEGORY";
3352 emitted_a_category |=
true;
3357 if (emitted_a_category)
3359 o <<
"HARMLESS_ENUM_CHANGE_CATEGORY";
3360 emitted_a_category |=
true;
3365 if (emitted_a_category)
3367 o <<
"HARMLESS_DATA_MEMBER_CHANGE_CATEGORY";
3368 emitted_a_category |=
true;
3373 if (emitted_a_category)
3375 o <<
"HARMLESS_SYMBOL_ALIAS_CHANGE_CATEGORY";
3376 emitted_a_category |=
true;
3381 if (emitted_a_category)
3383 o <<
"HARMLESS_UNION_OR_CLASS_CHANGE_CATEGORY";
3384 emitted_a_category |=
true;
3389 if (emitted_a_category)
3391 o <<
"SUPPRESSED_CATEGORY";
3392 emitted_a_category |=
true;
3397 if (emitted_a_category)
3399 o <<
"PRIVATE_TYPE_CATEGORY";
3400 emitted_a_category |=
true;
3405 if (emitted_a_category)
3407 o <<
"SIZE_OR_OFFSET_CHANGE_CATEGORY";
3408 emitted_a_category |=
true;
3413 if (emitted_a_category)
3415 o <<
"VIRTUAL_MEMBER_CHANGE_CATEGORY";
3416 emitted_a_category |=
true;
3419 if (c & REFERENCE_LVALUENESS_CHANGE_CATEGORY)
3421 if (emitted_a_category)
3423 o <<
"REFERENCE_LVALUENESS_CHANGE_CATEGORY";
3424 emitted_a_category |=
true;
3429 if (emitted_a_category)
3431 o <<
"NON_COMPATIBLE_DISTINCT_CHANGE_CATEGORY";
3432 emitted_a_category |=
true;
3437 if (emitted_a_category)
3439 o <<
"NON_COMPATIBLE_NAME_CHANGE_CATEGORY";
3440 emitted_a_category |=
true;
3445 if (emitted_a_category)
3447 o <<
"REDUNDANT_CATEGORY";
3448 emitted_a_category |=
true;
3453 if (emitted_a_category)
3455 o <<
"TYPE_DECL_ONLY_DEF_CHANGE_CATEGORY";
3456 emitted_a_category |=
true;
3461 if (emitted_a_category)
3463 o <<
"FN_PARM_TYPE_TOP_CV_CHANGE_CATEGORY";
3464 emitted_a_category |=
true;
3469 if (emitted_a_category)
3471 o <<
"FN_PARM_TYPE_CV_CHANGE_CATEGORY";
3472 emitted_a_category |=
true;
3477 if (emitted_a_category)
3479 o <<
"FN_RETURN_TYPE_CV_CHANGE_CATEGORY";
3480 emitted_a_category |=
true;
3485 if (emitted_a_category)
3487 o <<
"FN_PARM_ADD_REMOVE_CHANGE_CATEGORY";
3488 emitted_a_category |=
true;
3493 if (emitted_a_category)
3495 o <<
"VAR_TYPE_CV_CHANGE_CATEGORY";
3496 emitted_a_category |=
true;
3501 if (emitted_a_category)
3503 o <<
"VOID_PTR_TO_PTR_CHANGE_CATEGORY";
3504 emitted_a_category |=
true;
3509 if (emitted_a_category)
3511 o <<
"BENIGN_INFINITE_ARRAY_CHANGE_CATEGORY";
3512 emitted_a_category |=
true;
3517 if (emitted_a_category)
3519 o <<
"HAS_ALLOWED_CHANGE_CATEGORY";
3520 emitted_a_category |=
true;
3525 if (emitted_a_category)
3527 o <<
"HAS_DESCENDANT_WITH_ALLOWED_CHANGE_CATEGORY";
3528 emitted_a_category |=
true;
3533 if (emitted_a_category)
3535 o <<
"HAS_PARENT_WITH_ALLOWED_CHANGE_CATEGORY";
3536 emitted_a_category |=
true;
3559compute_diff_for_decls(
const decl_base_sptr first,
3560 const decl_base_sptr second,
3566 ((d = try_to_diff<function_decl>(first, second, ctxt))
3567 || (d = try_to_diff<var_decl>(first, second, ctxt))
3568 || (d = try_to_diff_distinct_kinds(first, second, ctxt)));
3591 const decl_base_sptr second,
3594 if (!first || !second)
3599 d = compute_diff_for_types(first, second, ctxt);
3601 d = compute_diff_for_decls(first, second, ctxt);
3621 const type_base_sptr second,
3627 diff_sptr d = compute_diff_for_types(f,s, ctxt);
3642 string prefix=
"diff of ";
3662 if (diff::priv_->pretty_representation_.empty())
3664 std::ostringstream o;
3670 diff::priv_->pretty_representation_ = o.str();
3672 return diff::priv_->pretty_representation_;
3715 if (
diff_sptr result = priv_->type_diff_.lock())
3722 context()->keep_diff_alive(result);
3723 priv_->type_diff_ = result;
3744 return ir::NO_CHANGE_KIND;
3756 context()->get_reporter()->report(*
this, out, indent);
3777 ctxt->initialize_canonical_diff(d);
3807 priv_(new
priv(underlying))
3815{
return dynamic_pointer_cast<pointer_type_def>(
first_subject());}
3822{
return dynamic_pointer_cast<pointer_type_def>(
second_subject());}
3829 if (diff::priv_->pretty_representation_.empty())
3831 std::ostringstream o;
3832 o <<
"pointer_diff["
3837 diff::priv_->pretty_representation_ = o.str();
3839 return diff::priv_->pretty_representation_;
3858 return ir::NO_CHANGE_KIND;
3867{
return priv_->underlying_type_diff_;}
3876{priv_->underlying_type_diff_ = d;}
3887 context()->get_reporter()->report(*
this, out, indent);
3907 diff_sptr d = compute_diff_for_types(first->get_pointed_to_type(),
3908 second->get_pointed_to_type(),
3911 ctxt->initialize_canonical_diff(result);
3936 priv_(new
priv(underlying_type_diff))
3964{
return priv_->underlying_type_diff_;}
3973 if (diff::priv_->pretty_representation_.empty())
3975 std::ostringstream o;
3976 o <<
"subrange_diff["
3981 diff::priv_->pretty_representation_ = o.str();
3983 return diff::priv_->pretty_representation_;
4005 return ir::NO_CHANGE_KIND;
4015{
context()->get_reporter()->report(*
this, out, indent);}
4043 diff_sptr d = compute_diff_for_types(first->get_underlying_type(),
4044 second->get_underlying_type(),
4048 ctxt->initialize_canonical_diff(result);
4083 vector<subrange_diff_sptr>& subrange_diffs,
4086 priv_(new
priv(element_type_diff, subrange_diffs))
4094{
return dynamic_pointer_cast<array_type_def>(
first_subject());}
4108{
return priv_->element_type_diff_;}
4113const vector<subrange_diff_sptr>&
4115{
return priv_->subrange_diffs_;}
4136{priv_->element_type_diff_ = d;}
4143{priv_->subrange_diffs_ = d;}
4150 if (diff::priv_->pretty_representation_.empty())
4152 std::ostringstream o;
4158 diff::priv_->pretty_representation_ = o.str();
4160 return diff::priv_->pretty_representation_;
4177 if (f->get_name() != s->get_name())
4179 if (f->get_size_in_bits() != s->get_size_in_bits())
4181 if (f->get_alignment_in_bits() != s->get_alignment_in_bits())
4204 return ir::NO_CHANGE_KIND;
4215 context()->get_reporter()->report(*
this, out, indent);
4233 diff_sptr element_diff = compute_diff_for_types(first->get_element_type(),
4234 second->get_element_type(),
4236 vector<subrange_diff_sptr> subrange_diffs;
4237 if (first->get_subranges().size() == first->get_subranges().size())
4239 for (
unsigned i = 0; i < first->get_subranges().size(); ++i)
4243 second->get_subranges()[i],
4252 ctxt->initialize_canonical_diff(result);
4280 priv_(new
priv(underlying))
4288{
return dynamic_pointer_cast<reference_type_def>(
first_subject());}
4295{
return dynamic_pointer_cast<reference_type_def>(
second_subject());}
4303{
return priv_->underlying_type_diff_;}
4311 priv_->underlying_type_diff_ = d;
4312 return priv_->underlying_type_diff_;
4320 if (diff::priv_->pretty_representation_.empty())
4322 std::ostringstream o;
4323 o <<
"reference_diff["
4328 diff::priv_->pretty_representation_ = o.str();
4330 return diff::priv_->pretty_representation_;
4351 return ir::NO_CHANGE_KIND;
4362 context()->get_reporter()->report(*
this, out, indent);
4380 diff_sptr d = compute_diff_for_types(first->get_pointed_to_type(),
4381 second->get_pointed_to_type(),
4384 ctxt->initialize_canonical_diff(result);
4411 priv_(new
priv(member_type_diff, containing_type_diff))
4421{
return dynamic_pointer_cast<ptr_to_mbr_type>(
first_subject());}
4439{
return priv_->member_type_diff_;}
4448{
return priv_->containing_type_diff_;}
4468 return ir::NO_CHANGE_KIND;
4478 if (diff::priv_->pretty_representation_.empty())
4480 std::ostringstream o;
4481 o <<
"ptr_to_mbr_diff["
4486 diff::priv_->pretty_representation_ = o.str();
4488 return diff::priv_->pretty_representation_;
4494 context()->get_reporter()->report(*
this, out, indent);
4531 is_type(second->get_member_type()),
4536 is_type(second->get_containing_type()),
4541 containing_type_diff,
4543 ctxt->initialize_canonical_diff(result);
4568 qualified_type_def_sptr second,
4572 priv_(new
priv(under))
4578const qualified_type_def_sptr
4580{
return dynamic_pointer_cast<qualified_type_def>(
first_subject());}
4585const qualified_type_def_sptr
4587{
return dynamic_pointer_cast<qualified_type_def>(
second_subject());}
4596{
return priv_->underlying_type_diff;}
4606 if (!priv_->leaf_underlying_type_diff)
4607 priv_->leaf_underlying_type_diff
4612 return priv_->leaf_underlying_type_diff;
4622{priv_->underlying_type_diff = d;}
4629 if (diff::priv_->pretty_representation_.empty())
4631 std::ostringstream o;
4632 o <<
"qualified_type_diff["
4637 diff::priv_->pretty_representation_ = o.str();
4639 return diff::priv_->pretty_representation_;
4658 return ir::NO_CHANGE_KIND;
4669 context()->get_reporter()->report(*
this, out, indent);
4682qualified_type_diff_sptr
4684 const qualified_type_def_sptr second,
4687 diff_sptr d = compute_diff_for_types(first->get_underlying_type(),
4688 second->get_underlying_type(),
4692 ctxt->initialize_canonical_diff(result);
4705enum_diff::clear_lookup_tables()
4707 priv_->deleted_enumerators_.clear();
4708 priv_->inserted_enumerators_.clear();
4709 priv_->changed_enumerators_.clear();
4716enum_diff::lookup_tables_empty()
const
4718 return (priv_->deleted_enumerators_.empty()
4719 && priv_->inserted_enumerators_.empty()
4720 && priv_->changed_enumerators_.empty());
4726enum_diff::ensure_lookup_tables_populated()
4728 if (!lookup_tables_empty())
4732 edit_script e = priv_->enumerators_changes_;
4734 for (vector<deletion>::const_iterator it = e.deletions().begin();
4735 it != e.deletions().end();
4738 unsigned i = it->index();
4739 const enum_type_decl::enumerator& n =
4741 const string& name = n.get_name();
4742 ABG_ASSERT(priv_->deleted_enumerators_.find(n.get_name())
4743 == priv_->deleted_enumerators_.end());
4744 priv_->deleted_enumerators_[name] = n;
4747 for (vector<insertion>::const_iterator it = e.insertions().begin();
4748 it != e.insertions().end();
4751 for (vector<unsigned>::const_iterator iit =
4752 it->inserted_indexes().begin();
4753 iit != it->inserted_indexes().end();
4757 const enum_type_decl::enumerator& n =
4759 const string& name = n.get_name();
4760 ABG_ASSERT(priv_->inserted_enumerators_.find(n.get_name())
4761 == priv_->inserted_enumerators_.end());
4762 string_enumerator_map::const_iterator j =
4763 priv_->deleted_enumerators_.find(name);
4764 if (j == priv_->deleted_enumerators_.end())
4765 priv_->inserted_enumerators_[name] = n;
4769 priv_->changed_enumerators_[j->first] =
4770 std::make_pair(j->second, n);
4771 priv_->deleted_enumerators_.erase(j);
4784 for (
auto& entry : priv_->inserted_enumerators_)
4786 enum_type_decl::enumerator& final_enumerator = entry.second;
4787 enum_type_decl::enumerator initial_enumerator;
4788 if (
first_enum()->find_enumerator_by_value(entry.second.get_value(),
4789 initial_enumerator))
4791 enum_type_decl::enumerator foo;
4793 find_enumerator_by_name(initial_enumerator.get_name(),
4796 priv_->changed_enumerators_[initial_enumerator.get_name()] =
4797 std::make_pair(initial_enumerator, final_enumerator);
4798 enums_to_erase.push_back(final_enumerator);
4803 for (
auto& enomerator : enums_to_erase)
4804 priv_->inserted_enumerators_.erase(enomerator.get_name());
4813 enums_to_erase.clear();
4814 for (
auto& entry : priv_->deleted_enumerators_)
4816 enum_type_decl::enumerator& initial_enumerator = entry.second;
4817 enum_type_decl::enumerator final_enumerator;
4818 if (
second_enum()->find_enumerator_by_value(entry.second.get_value(),
4821 enum_type_decl::enumerator foo;
4823 find_enumerator_by_name(final_enumerator.get_name(),
4826 priv_->changed_enumerators_[initial_enumerator.get_name()] =
4827 std::make_pair(initial_enumerator, final_enumerator);
4828 enums_to_erase.push_back(initial_enumerator);
4833 for (
auto& enomerator : enums_to_erase)
4834 priv_->deleted_enumerators_.erase(enomerator.get_name());
4862 priv_(new
priv(underlying_type_diff))
4868{
return dynamic_pointer_cast<enum_type_decl>(
first_subject());}
4878{
return priv_->underlying_type_diff_;}
4883{
return priv_->deleted_enumerators_;}
4888{
return priv_->inserted_enumerators_;}
4893{
return priv_->changed_enumerators_;}
4900 if (diff::priv_->pretty_representation_.empty())
4902 std::ostringstream o;
4908 diff::priv_->pretty_representation_ = o.str();
4910 return diff::priv_->pretty_representation_;
4929 return ir::NO_CHANGE_KIND;
4940 context()->get_reporter()->report(*
this, out, indent);
4962 diff_sptr ud = compute_diff_for_types(first->get_underlying_type(),
4963 second->get_underlying_type(),
4965 enum_diff_sptr d(
new enum_diff(first, second, ud, ctxt));
4966 if (first != second)
4969 first->get_enumerators().end(),
4970 second->get_enumerators().begin(),
4971 second->get_enumerators().end(),
4972 d->priv_->enumerators_changes_);
4973 d->ensure_lookup_tables_populated();
4975 ctxt->initialize_canonical_diff(d);
4997 string qname = d->get_qualified_name();
4998 string_diff_sptr_map::const_iterator it =
4999 changed_member_types_.find(qname);
5001 return ((it == changed_member_types_.end())
5003 : it->second->second_subject());
5020 string qname = d->get_qualified_name();
5021 string_var_diff_sptr_map::const_iterator it =
5022 subtype_changed_dm_.find(qname);
5024 if (it == subtype_changed_dm_.end())
5025 return decl_base_sptr();
5026 return it->second->second_var();
5044 string qname = d->get_qualified_name();
5045 string_diff_sptr_map::const_iterator it =
5046 changed_member_class_tmpls_.find(qname);
5048 return ((it == changed_member_class_tmpls_.end())
5050 : dynamic_pointer_cast<decl_base>(it->second->second_subject()));
5061 for (string_decl_base_sptr_map::const_iterator i =
5062 deleted_data_members_.begin();
5063 i != deleted_data_members_.end();
5080 for (string_decl_base_sptr_map::const_iterator i =
5081 inserted_data_members_.begin();
5082 i != inserted_data_members_.end();
5102 size_t num_filtered= 0;
5103 for (var_diff_sptrs_type::const_iterator i =
5104 sorted_subtype_changed_dm_.begin();
5105 i != sorted_subtype_changed_dm_.end();
5110 if ((*i)->has_local_changes() && (*i)->is_filtered_out())
5115 if ((*i)->is_filtered_out())
5119 return num_filtered;
5133 size_t num_filtered= 0;
5135 for (unsigned_var_diff_sptr_map::const_iterator i = changed_dm_.begin();
5136 i != changed_dm_.end();
5151 return num_filtered;
5160#define SKIP_MEM_FN_IF_VIRTUALITY_DISALLOWED \
5162 if (get_member_function_is_virtual(f) \
5163 || get_member_function_is_virtual(s)) \
5165 if (!(allowed_category | VIRTUAL_MEMBER_CHANGE_CATEGORY)) \
5170 if (!(allowed_category | NON_VIRT_MEM_FUN_CHANGE_CATEGORY)) \
5185 diff_category allowed_category = ctxt->get_allowed_category();
5187 for (function_decl_diff_sptrs_type::const_iterator i =
5188 sorted_changed_member_functions_.begin();
5189 i != sorted_changed_member_functions_.end();
5192 method_decl_sptr f =
5193 dynamic_pointer_cast<method_decl>
5194 ((*i)->first_function_decl());
5197 method_decl_sptr s =
5198 dynamic_pointer_cast<method_decl>
5199 ((*i)->second_function_decl());
5205 ctxt->maybe_apply_filters(
diff);
5224 diff_category allowed_category = ctxt->get_allowed_category();
5226 for (string_member_function_sptr_map::const_iterator i =
5227 inserted_member_functions_.begin();
5228 i != inserted_member_functions_.end();
5231 method_decl_sptr f = i->second,
5237 ctxt->maybe_apply_filters(
diff);
5257 diff_category allowed_category = ctxt->get_allowed_category();
5259 for (string_member_function_sptr_map::const_iterator i =
5260 deleted_member_functions_.begin();
5261 i != deleted_member_functions_.end();
5264 method_decl_sptr f = i->second,
5270 ctxt->maybe_apply_filters(
diff);
5287 priv_->deleted_member_types_.clear();
5288 priv_->inserted_member_types_.clear();
5289 priv_->changed_member_types_.clear();
5290 priv_->deleted_data_members_.clear();
5291 priv_->inserted_data_members_.clear();
5292 priv_->subtype_changed_dm_.clear();
5293 priv_->deleted_member_functions_.clear();
5294 priv_->inserted_member_functions_.clear();
5295 priv_->changed_member_functions_.clear();
5296 priv_->deleted_member_class_tmpls_.clear();
5297 priv_->inserted_member_class_tmpls_.clear();
5298 priv_->changed_member_class_tmpls_.clear();
5307 return (priv_->deleted_member_types_.empty()
5308 && priv_->inserted_member_types_.empty()
5309 && priv_->changed_member_types_.empty()
5310 && priv_->deleted_data_members_.empty()
5311 && priv_->inserted_data_members_.empty()
5312 && priv_->subtype_changed_dm_.empty()
5313 && priv_->inserted_member_functions_.empty()
5314 && priv_->deleted_member_functions_.empty()
5315 && priv_->changed_member_functions_.empty()
5316 && priv_->deleted_member_class_tmpls_.empty()
5317 && priv_->inserted_member_class_tmpls_.empty()
5318 && priv_->changed_member_class_tmpls_.empty());
5329 for (vector<deletion>::const_iterator it = e.deletions().begin();
5330 it != e.deletions().end();
5333 unsigned i = it->index();
5337 if (record_type && record_type->get_is_declaration_only())
5339 string name = d->get_name();
5340 priv_->deleted_member_types_[name] = d;
5343 for (vector<insertion>::const_iterator it = e.insertions().begin();
5344 it != e.insertions().end();
5347 for (vector<unsigned>::const_iterator iit =
5348 it->inserted_indexes().begin();
5349 iit != it->inserted_indexes().end();
5356 if (record_type && record_type->get_is_declaration_only())
5358 string name = d->get_name();
5359 string_decl_base_sptr_map::const_iterator j =
5360 priv_->deleted_member_types_.find(name);
5361 if (j != priv_->deleted_member_types_.end())
5363 if (*j->second != *d)
5364 priv_->changed_member_types_[name] =
5367 priv_->deleted_member_types_.erase(j);
5370 priv_->inserted_member_types_[name] = d;
5378 for (vector<deletion>::const_iterator it = e.deletions().begin();
5379 it != e.deletions().end();
5382 unsigned i = it->index();
5385 string name = data_member->get_anon_dm_reliable_name();
5387 ABG_ASSERT(priv_->deleted_data_members_.find(name)
5388 == priv_->deleted_data_members_.end());
5389 priv_->deleted_data_members_[name] = data_member;
5392 for (vector<insertion>::const_iterator it = e.insertions().begin();
5393 it != e.insertions().end();
5396 for (vector<unsigned>::const_iterator iit =
5397 it->inserted_indexes().begin();
5398 iit != it->inserted_indexes().end();
5405 string name = added_dm->get_anon_dm_reliable_name();
5406 ABG_ASSERT(priv_->inserted_data_members_.find(name)
5407 == priv_->inserted_data_members_.end());
5409 bool ignore_added_anonymous_data_member =
false;
5447 bool added_anon_dm_changes_dm =
false;
5450 vector<var_decl_sptr> dms_replaced_by_anon_dm;
5459 for (string_decl_base_sptr_map::const_iterator it =
5460 priv_->deleted_data_members_.begin();
5461 it != priv_->deleted_data_members_.end();
5470 string deleted_dm_name = it->second->get_name();
5486 size_t replaced_dm_offset =
5488 replacing_dm_offset =
5491 if (replaced_dm_offset != replacing_dm_offset)
5499 added_anon_dm_changes_dm =
true;
5503 if (replaced_dm->get_type()->get_size_in_bits()
5504 == replaced_dm->get_type()->get_size_in_bits())
5505 dms_replaced_by_anon_dm.push_back(replaced_dm);
5508 added_anon_dm_changes_dm =
true;
5517 if (!added_anon_dm_changes_dm
5518 && !dms_replaced_by_anon_dm.empty())
5521 type_base_sptr added_dm_type = added_dm->get_type();
5537 ignore_added_anonymous_data_member =
true;
5538 for (vector<var_decl_sptr>::const_iterator i =
5539 dms_replaced_by_anon_dm.begin();
5540 i != dms_replaced_by_anon_dm.end();
5543 string n = (*i)->get_name();
5544 priv_->dms_replaced_by_adms_[n] =
5546 priv_->deleted_data_members_.erase(n);
5552 if (!ignore_added_anonymous_data_member)
5565 string_decl_base_sptr_map::const_iterator j =
5566 priv_->deleted_data_members_.find(name);
5567 if (j != priv_->deleted_data_members_.end())
5569 if (*j->second != *d)
5572 priv_->subtype_changed_dm_[name]=
5575 priv_->deleted_data_members_.erase(j);
5578 priv_->inserted_data_members_[name] = d;
5586 for (string_decl_base_sptr_map::const_iterator i =
5587 priv_->deleted_data_members_.begin();
5588 i != priv_->deleted_data_members_.end();
5592 priv_->deleted_dm_by_offset_[offset] = i->second;
5595 for (string_decl_base_sptr_map::const_iterator i =
5596 priv_->inserted_data_members_.begin();
5597 i != priv_->inserted_data_members_.end();
5601 priv_->inserted_dm_by_offset_[offset] = i->second;
5604 for (unsigned_decl_base_sptr_map::const_iterator i =
5605 priv_->inserted_dm_by_offset_.begin();
5606 i != priv_->inserted_dm_by_offset_.end();
5609 unsigned_decl_base_sptr_map::const_iterator j =
5610 priv_->deleted_dm_by_offset_.find(i->first);
5611 if (j != priv_->deleted_dm_by_offset_.end())
5615 priv_->changed_dm_[i->first] =
5620 for (unsigned_var_diff_sptr_map::const_iterator i =
5621 priv_->changed_dm_.begin();
5622 i != priv_->changed_dm_.end();
5625 priv_->deleted_dm_by_offset_.erase(i->first);
5626 priv_->inserted_dm_by_offset_.erase(i->first);
5627 priv_->deleted_data_members_.erase
5628 (i->second->first_var()->get_anon_dm_reliable_name());
5629 priv_->inserted_data_members_.erase
5630 (i->second->second_var()->get_anon_dm_reliable_name());
5638 non_anonymous_dms_in_second_class);
5639 vector<string> deleted_data_members_to_delete;
5641 for (
auto& entry : priv_->deleted_data_members_)
5658 bool anonymous_dm_members_present =
true;
5662 for (
auto& e : non_anonymous_data_members)
5664 if (non_anonymous_dms_in_second_class.find(e.first)
5665 == non_anonymous_dms_in_second_class.end())
5670 anonymous_dm_members_present =
false;
5672 if (anonymous_dm_members_present)
5677 deleted_data_members_to_delete.push_back(data_member->get_anon_dm_reliable_name());
5683 for (
string& name_of_dm_to_delete: deleted_data_members_to_delete)
5684 priv_->deleted_data_members_.erase(name_of_dm_to_delete);
5687 priv_->sorted_subtype_changed_dm_);
5689 priv_->sorted_changed_dm_);
5692 edit_script& e = priv_->member_class_tmpls_changes_;
5694 for (vector<deletion>::const_iterator it = e.deletions().begin();
5695 it != e.deletions().end();
5698 unsigned i = it->index();
5702 string name = d->get_name();
5703 ABG_ASSERT(priv_->deleted_member_class_tmpls_.find(name)
5704 == priv_->deleted_member_class_tmpls_.end());
5705 priv_->deleted_member_class_tmpls_[name] = d;
5708 for (vector<insertion>::const_iterator it = e.insertions().begin();
5709 it != e.insertions().end();
5712 for (vector<unsigned>::const_iterator iit =
5713 it->inserted_indexes().begin();
5714 iit != it->inserted_indexes().end();
5721 string name = d->get_name();
5722 ABG_ASSERT(priv_->inserted_member_class_tmpls_.find(name)
5723 == priv_->inserted_member_class_tmpls_.end());
5724 string_decl_base_sptr_map::const_iterator j =
5725 priv_->deleted_member_class_tmpls_.find(name);
5726 if (j != priv_->deleted_member_class_tmpls_.end())
5728 if (*j->second != *d)
5729 priv_->changed_member_types_[name]=
5731 priv_->deleted_member_class_tmpls_.erase(j);
5734 priv_->inserted_member_class_tmpls_[name] = d;
5739 priv_->sorted_changed_member_types_);
5748 priv_.reset(
new priv);
5759 class_or_union_sptr second_scope,
5777const class_or_union_diff::priv_ptr&
5790 return canonical->priv_;
5812{
return get_priv()->member_types_changes_;}
5818{
return get_priv()->member_types_changes_;}
5824{
return get_priv()->data_members_changes_;}
5830{
return get_priv()->data_members_changes_;}
5837{
return get_priv()->inserted_data_members_;}
5844{
return get_priv()->deleted_data_members_;}
5850{
return get_priv()->member_fns_changes_;}
5859{
return get_priv()->sorted_changed_member_functions_;}
5865{
return get_priv()->member_fns_changes_;}
5870{
return get_priv()->deleted_member_functions_;}
5875{
return get_priv()->inserted_member_functions_;}
5893{
return get_priv()->sorted_changed_dm_;}
5901{
return get_priv()->count_filtered_changed_dm(local);}
5908{
return get_priv()->sorted_subtype_changed_dm_;}
5915{
return get_priv()->count_filtered_subtype_changed_dm(local);}
5927{
return get_priv()->dms_replaced_by_adms_;}
5939 if (priv_->dms_replaced_by_adms_ordered_.empty())
5941 for (string_decl_base_sptr_map::const_iterator it =
5942 priv_->dms_replaced_by_adms_.begin();
5943 it != priv_->dms_replaced_by_adms_.end();
5950 priv_->dms_replaced_by_adms_ordered_.push_back(changed_dm);
5955 return priv_->dms_replaced_by_adms_ordered_;
5962{
return get_priv()->member_fn_tmpls_changes_;}
5968{
return get_priv()->member_fn_tmpls_changes_;}
5974{
return get_priv()->member_class_tmpls_changes_;}
5980{
return get_priv()->member_class_tmpls_changes_;}
5996 return ir::NO_CHANGE_KIND;
6009 context()->get_reporter()->report(*
this, out, indent);
6021 for (var_diff_sptrs_type::const_iterator i =
6022 get_priv()->sorted_subtype_changed_dm_.begin();
6023 i !=
get_priv()->sorted_subtype_changed_dm_.end();
6028 for (var_diff_sptrs_type::const_iterator i =
6029 get_priv()->sorted_changed_dm_.begin();
6030 i !=
get_priv()->sorted_changed_dm_.end();
6036 for (diff_sptrs_type::const_iterator i =
6037 get_priv()->sorted_changed_member_types_.begin();
6038 i !=
get_priv()->sorted_changed_member_types_.end();
6044 for (function_decl_diff_sptrs_type::const_iterator i =
6045 get_priv()->sorted_changed_member_functions_.begin();
6046 i !=
get_priv()->sorted_changed_member_functions_.end();
6061class_diff::clear_lookup_tables(
void)
6063 priv_->deleted_bases_.clear();
6064 priv_->inserted_bases_.clear();
6065 priv_->changed_bases_.clear();
6072class_diff::lookup_tables_empty(
void)
const
6074 return (priv_->deleted_bases_.empty()
6075 && priv_->inserted_bases_.empty()
6076 && priv_->changed_bases_.empty());
6086static string_member_function_sptr_map::const_iterator
6089 for (string_member_function_sptr_map::const_iterator i = map.begin();
6103class_diff::ensure_lookup_tables_populated(
void)
const
6107 if (!lookup_tables_empty())
6111 edit_script& e = get_priv()->base_changes_;
6113 for (vector<deletion>::const_iterator it = e.deletions().begin();
6114 it != e.deletions().end();
6117 unsigned i = it->index();
6120 string name = b->get_base_class()->get_qualified_name();
6121 ABG_ASSERT(get_priv()->deleted_bases_.find(name)
6122 == get_priv()->deleted_bases_.end());
6123 get_priv()->deleted_bases_[name] = b;
6126 for (vector<insertion>::const_iterator it = e.insertions().begin();
6127 it != e.insertions().end();
6130 for (vector<unsigned>::const_iterator iit =
6131 it->inserted_indexes().begin();
6132 iit != it->inserted_indexes().end();
6138 string name = b->get_base_class()->get_qualified_name();
6139 ABG_ASSERT(get_priv()->inserted_bases_.find(name)
6140 == get_priv()->inserted_bases_.end());
6141 string_base_sptr_map::const_iterator j =
6142 get_priv()->deleted_bases_.find(name);
6143 if (j != get_priv()->deleted_bases_.end())
6146 get_priv()->changed_bases_[name] =
6152 get_priv()->moved_bases_.push_back(b);
6153 get_priv()->deleted_bases_.erase(j);
6156 get_priv()->inserted_bases_[name] = b;
6166 class_or_union_diff::priv_->deleted_data_members_;
6168 vector<var_decl_sptr> deleted_data_members_present_in_bases;
6176 var_decl_sptr member = klass->find_data_member(deleted_member->get_name());
6178 deleted_data_members_present_in_bases.push_back(member);
6185 for (
var_decl_sptr m : deleted_data_members_present_in_bases)
6187 string name = m->get_name();
6191 if (*deleted_member != *m)
6195 class_or_union_diff::priv_->subtype_changed_dm_[name]= dif;
6202 get_priv()->sorted_deleted_bases_);
6204 get_priv()->sorted_inserted_bases_);
6206 get_priv()->sorted_changed_bases_);
6211 edit_script& e = p->member_fns_changes_;
6213 for (vector<deletion>::const_iterator it = e.deletions().begin();
6214 it != e.deletions().end();
6217 unsigned i = it->index();
6218 method_decl_sptr mem_fn =
6220 string name = mem_fn->get_linkage_name();
6222 name = mem_fn->get_pretty_representation();
6224 if (p->deleted_member_functions_.find(name)
6225 != p->deleted_member_functions_.end())
6227 p->deleted_member_functions_[name] = mem_fn;
6230 for (vector<insertion>::const_iterator it = e.insertions().begin();
6231 it != e.insertions().end();
6234 for (vector<unsigned>::const_iterator iit =
6235 it->inserted_indexes().begin();
6236 iit != it->inserted_indexes().end();
6241 method_decl_sptr mem_fn =
6243 string name = mem_fn->get_linkage_name();
6245 name = mem_fn->get_pretty_representation();
6247 if (p->inserted_member_functions_.find(name)
6248 != p->inserted_member_functions_.end())
6250 string_member_function_sptr_map::const_iterator j =
6251 p->deleted_member_functions_.find(name);
6253 if (j != p->deleted_member_functions_.end())
6255 if (*j->second != *mem_fn)
6256 p->changed_member_functions_[name] =
6257 compute_diff(static_pointer_cast<function_decl>(j->second),
6258 static_pointer_cast<function_decl>(mem_fn),
6260 p->deleted_member_functions_.erase(j);
6263 p->inserted_member_functions_[name] = mem_fn;
6310 vector<string> to_delete;
6311 corpus_sptr f =
context()->get_first_corpus(),
6312 s =
context()->get_second_corpus();
6314 for (string_member_function_sptr_map::const_iterator i =
6333 find_virtual_dtor_in_map(p->inserted_member_functions_);
6334 if (it != p->inserted_member_functions_.end())
6342 (!i->second->get_linkage_name().empty())
6343 ? i->second->get_linkage_name()
6344 : i->second->get_pretty_representation();
6345 to_delete.push_back(name);
6346 p->inserted_member_functions_.erase(it);
6353 if (!i->second->get_symbol()
6354 || s->lookup_function_symbol(*i->second->get_symbol()))
6355 to_delete.push_back(i->first);
6359 for (vector<string>::const_iterator i = to_delete.begin();
6360 i != to_delete.end();
6362 p->deleted_member_functions_.erase(*i);
6367 for (string_member_function_sptr_map::const_iterator i =
6376 if (!i->second->get_symbol()
6377 || f->lookup_function_symbol(*i->second->get_symbol()))
6378 to_delete.push_back(i->first);
6381 for (vector<string>::const_iterator i = to_delete.begin();
6382 i != to_delete.end();
6384 p->inserted_member_functions_.erase(*i);
6387 p->sorted_deleted_member_functions_);
6390 p->sorted_inserted_member_functions_);
6393 (p->changed_member_functions_,
6394 p->sorted_changed_member_functions_);
6401class_diff::allocate_priv_data()
6405 priv_.reset(
new priv);
6418 string qname = d->get_base_class()->get_qualified_name();
6419 string_base_diff_sptr_map::const_iterator it =
6420 changed_bases_.find(qname);
6422 return (it == changed_bases_.end())
6424 : it->second->second_base();
6435 size_t num_filtered = 0;
6436 for (base_diff_sptrs_type::const_iterator i = sorted_changed_bases_.begin();
6437 i != sorted_changed_bases_.end();
6444 return num_filtered;
6458 for (base_diff_sptrs_type::const_iterator i =
6459 get_priv()->sorted_changed_bases_.begin();
6460 i != get_priv()->sorted_changed_bases_.end();
6485class_diff::~class_diff()
6499const class_diff::priv_ptr&
6500class_diff::get_priv()
const
6512 return canonical->priv_;
6520 if (diff::priv_->pretty_representation_.empty())
6522 std::ostringstream o;
6528 diff::priv_->pretty_representation_ = o.str();
6530 return diff::priv_->pretty_representation_;
6549 return ir::NO_CHANGE_KIND;
6553shared_ptr<class_decl>
6560shared_ptr<class_decl>
6567{
return get_priv()->base_changes_;}
6575{
return get_priv()->deleted_bases_;}
6583{
return get_priv()->inserted_bases_;}
6590{
return get_priv()->sorted_changed_bases_;}
6598const vector<class_decl::base_spec_sptr>&
6600{
return get_priv()->moved_bases_;}
6605{
return get_priv()->base_changes_;}
6616 context()->get_reporter()->report(*
this, out, indent);
6641 ctxt->initialize_canonical_diff(changes);
6644 if (!ctxt->get_canonical_diff_for(first, second))
6648 diff_sptr canonical_diff = ctxt->get_canonical_diff_for(changes);
6650 ctxt->set_canonical_diff_for(first, second, canonical_diff);
6666 if (
is_class_diff(changes->get_canonical_diff()) == changes.get())
6669 changes->allocate_priv_data();
6681 f->get_base_specifiers().end(),
6682 s->get_base_specifiers().begin(),
6683 s->get_base_specifiers().end(),
6684 changes->base_changes());
6690 f->get_member_types().end(),
6691 s->get_member_types().begin(),
6692 s->get_member_types().end(),
6693 changes->member_types_changes());
6698 f->get_non_static_data_members().end(),
6699 s->get_non_static_data_members().begin(),
6700 s->get_non_static_data_members().end(),
6701 changes->data_members_changes());
6705 f->get_virtual_mem_fns().end(),
6706 s->get_virtual_mem_fns().begin(),
6707 s->get_virtual_mem_fns().end(),
6708 changes->member_fns_changes());
6711 compute_diff(f->get_member_function_templates().begin(),
6712 f->get_member_function_templates().end(),
6713 s->get_member_function_templates().begin(),
6714 s->get_member_function_templates().end(),
6715 changes->member_fn_tmpls_changes());
6720 f->get_member_class_templates().end(),
6721 s->get_member_class_templates().begin(),
6722 s->get_member_class_templates().end(),
6723 changes->member_class_tmpls_changes());
6726 changes->ensure_lookup_tables_populated();
6756 :
diff(first, second, ctxt),
6757 priv_(new
priv(underlying))
6765{
return dynamic_pointer_cast<class_decl::base_spec>(
first_subject());}
6772{
return dynamic_pointer_cast<class_decl::base_spec>(
second_subject());}
6781{
return priv_->underlying_class_diff_;}
6790{priv_->underlying_class_diff_ = d;}
6797 if (diff::priv_->pretty_representation_.empty())
6799 std::ostringstream o;
6805 diff::priv_->pretty_representation_ = o.str();
6807 return diff::priv_->pretty_representation_;
6826 return ir::NO_CHANGE_KIND;
6837 context()->get_reporter()->report(*
this, out, indent);
6859 second->get_base_class(),
6863 ctxt->initialize_canonical_diff(changes);
6878union_diff::clear_lookup_tables(
void)
6885union_diff::lookup_tables_empty(
void)
const
6891union_diff::ensure_lookup_tables_populated(
void)
const
6897union_diff::allocate_priv_data()
6910 union_decl_sptr second_union,
6933 if (diff::priv_->pretty_representation_.empty())
6935 std::ostringstream o;
6941 diff::priv_->pretty_representation_ = o.str();
6943 return diff::priv_->pretty_representation_;
6955 context()->get_reporter()->report(*
this, out, indent);
6970 const union_decl_sptr second,
6973 union_diff_sptr changes(
new union_diff(first, second, ctxt));
6975 ctxt->initialize_canonical_diff(changes);
6991 if (
is_union_diff(changes->get_canonical_diff()) == changes.get())
6994 changes->allocate_priv_data();
7005 compute_diff(first->get_non_static_data_members().begin(),
7006 first->get_non_static_data_members().end(),
7007 second->get_non_static_data_members().begin(),
7008 second->get_non_static_data_members().end(),
7009 changes->data_members_changes());
7014 first->get_mem_fns().end(),
7015 second->get_mem_fns().begin(),
7016 second->get_mem_fns().end(),
7017 changes->member_fns_changes());
7020 compute_diff(first->get_member_function_templates().begin(),
7021 first->get_member_function_templates().end(),
7022 second->get_member_function_templates().begin(),
7023 second->get_member_function_templates().end(),
7024 changes->member_fn_tmpls_changes());
7027 changes->ensure_lookup_tables_populated();
7041scope_diff::clear_lookup_tables()
7043 priv_->deleted_types_.clear();
7044 priv_->deleted_decls_.clear();
7045 priv_->inserted_types_.clear();
7046 priv_->inserted_decls_.clear();
7047 priv_->changed_types_.clear();
7048 priv_->changed_decls_.clear();
7049 priv_->removed_types_.clear();
7050 priv_->removed_decls_.clear();
7051 priv_->added_types_.clear();
7052 priv_->added_decls_.clear();
7062scope_diff::lookup_tables_empty()
const
7064 return (priv_->deleted_types_.empty()
7065 && priv_->deleted_decls_.empty()
7066 && priv_->inserted_types_.empty()
7067 && priv_->inserted_decls_.empty()
7068 && priv_->changed_types_.empty()
7069 && priv_->changed_decls_.empty()
7070 && priv_->removed_types_.empty()
7071 && priv_->removed_decls_.empty()
7072 && priv_->added_types_.empty()
7073 && priv_->added_decls_.empty());
7079scope_diff::ensure_lookup_tables_populated()
7081 if (!lookup_tables_empty())
7084 edit_script& e = priv_->member_changes_;
7087 for (
const auto& deletion : e.deletions())
7089 unsigned i = deletion.index();
7091 string qname = decl->get_qualified_name();
7095 if (klass_decl && klass_decl->get_is_declaration_only())
7106 == priv_->deleted_types_.end());
7107 priv_->deleted_types_[qname] = decl;
7112 == priv_->deleted_decls_.end());
7113 priv_->deleted_decls_[qname] = decl;
7119 for (vector<insertion>::const_iterator it = e.insertions().begin();
7120 it != e.insertions().end();
7123 for (vector<unsigned>::const_iterator i = it->inserted_indexes().begin();
7124 i != it->inserted_indexes().end();
7128 string qname = decl->get_qualified_name();
7132 dynamic_pointer_cast<class_decl>(decl);
7133 if (klass_decl && klass_decl->get_is_declaration_only())
7143 ABG_ASSERT(priv_->inserted_types_.find(qname)
7144 == priv_->inserted_types_.end());
7145 string_decl_base_sptr_map::const_iterator j =
7146 priv_->deleted_types_.find(qname);
7147 if (j != priv_->deleted_types_.end())
7149 if (*j->second != *decl)
7150 priv_->changed_types_[qname] =
7152 priv_->deleted_types_.erase(j);
7155 priv_->inserted_types_[qname] = decl;
7159 ABG_ASSERT(priv_->inserted_decls_.find(qname)
7160 == priv_->inserted_decls_.end());
7161 string_decl_base_sptr_map::const_iterator j =
7162 priv_->deleted_decls_.find(qname);
7163 if (j != priv_->deleted_decls_.end())
7165 if (*j->second != *decl)
7166 priv_->changed_decls_[qname] =
7168 priv_->deleted_decls_.erase(j);
7171 priv_->inserted_decls_[qname] = decl;
7177 priv_->sorted_changed_decls_);
7179 priv_->sorted_changed_types_);
7182 for (string_decl_base_sptr_map::const_iterator i =
7183 priv_->deleted_types_.begin();
7184 i != priv_->deleted_types_.end();
7187 string_decl_base_sptr_map::const_iterator r =
7188 priv_->inserted_types_.find(i->first);
7189 if (r == priv_->inserted_types_.end())
7190 priv_->removed_types_[i->first] = i->second;
7192 for (string_decl_base_sptr_map::const_iterator i =
7193 priv_->deleted_decls_.begin();
7194 i != priv_->deleted_decls_.end();
7197 string_decl_base_sptr_map::const_iterator r =
7198 priv_->inserted_decls_.find(i->first);
7199 if (r == priv_->inserted_decls_.end())
7200 priv_->removed_decls_[i->first] = i->second;
7204 for (string_decl_base_sptr_map::const_iterator i =
7205 priv_->inserted_types_.begin();
7206 i != priv_->inserted_types_.end();
7209 string_decl_base_sptr_map::const_iterator r =
7210 priv_->deleted_types_.find(i->first);
7211 if (r == priv_->deleted_types_.end())
7212 priv_->added_types_[i->first] = i->second;
7214 for (string_decl_base_sptr_map::const_iterator i =
7215 priv_->inserted_decls_.begin();
7216 i != priv_->inserted_decls_.end();
7219 string_decl_base_sptr_map::const_iterator r =
7220 priv_->deleted_decls_.find(i->first);
7221 if (r == priv_->deleted_decls_.end())
7222 priv_->added_decls_[i->first] = i->second;
7234 for (diff_sptrs_type::const_iterator i =
changed_types().begin();
7240 for (diff_sptrs_type::const_iterator i =
changed_decls().begin();
7260 :
diff(first_scope, second_scope, ctxt),
7298{
return priv_->member_changes_;}
7320{
return priv_->member_changes_;}
7335 return scope->get_member_decls()[i];
7366 return scope->get_member_decls()[i];
7387{
return priv_->sorted_changed_types_;}
7393{
return priv_->sorted_changed_decls_;}
7396scope_diff::removed_types()
const
7397{
return priv_->removed_types_;}
7400scope_diff::removed_decls()
const
7401{
return priv_->removed_decls_;}
7404scope_diff::added_types()
const
7405{
return priv_->added_types_;}
7408scope_diff::added_decls()
const
7409{
return priv_->added_decls_;}
7416 if (diff::priv_->pretty_representation_.empty())
7418 std::ostringstream o;
7424 diff::priv_->pretty_representation_ = o.str();
7426 return diff::priv_->pretty_representation_;
7448 return ir::NO_CHANGE_KIND;
7459 context()->get_reporter()->report(*
this, out, indent);
7485 ABG_ASSERT(d->first_scope() == first && d->second_scope() == second);
7488 first->get_member_decls().end(),
7489 second->get_member_decls().begin(),
7490 second->get_member_decls().end(),
7491 d->member_changes());
7493 d->ensure_lookup_tables_populated();
7519 ctxt->initialize_canonical_diff(d);
7543 ABG_ASSERT(first->get_index() == second->get_index());
7556{
return dynamic_pointer_cast<function_decl::parameter>(
first_subject());}
7564{
return dynamic_pointer_cast<function_decl::parameter>(
second_subject());}
7574{
return priv_->type_diff;}
7584 if (diff::priv_->pretty_representation_.empty())
7586 std::ostringstream o;
7587 o <<
"function_parameter_diff["
7592 diff::priv_->pretty_representation_ = o.str();
7594 return diff::priv_->pretty_representation_;
7615 return ir::NO_CHANGE_KIND;
7626 context()->get_reporter()->report(*
this, out, indent);
7660 if (!first || !second)
7664 ctxt->initialize_canonical_diff(result);
7673function_type_diff::ensure_lookup_tables_populated()
7675 priv_->return_type_diff_ =
7682 for (vector<deletion>::const_iterator i =
7683 priv_->parm_changes_.deletions().begin();
7684 i != priv_->parm_changes_.deletions().end();
7689 parm_name = parm->get_name_id();
7693 priv_->deleted_parms_[parm_name] = parm;
7694 priv_->deleted_parms_by_id_[parm->get_index()] = parm;
7697 for (vector<insertion>::const_iterator i =
7698 priv_->parm_changes_.insertions().begin();
7699 i != priv_->parm_changes_.insertions().end();
7702 for (vector<unsigned>::const_iterator j =
7703 i->inserted_indexes().begin();
7704 j != i->inserted_indexes().end();
7708 parm_name = parm->get_name_id();
7713 string_parm_map::const_iterator k =
7714 priv_->deleted_parms_.find(parm_name);
7715 if (k != priv_->deleted_parms_.end())
7717 if (*k->second != *parm)
7718 priv_->subtype_changed_parms_[parm_name] =
7720 priv_->deleted_parms_.erase(parm_name);
7723 priv_->added_parms_[parm_name] = parm;
7726 unsigned_parm_map::const_iterator k =
7727 priv_->deleted_parms_by_id_.find(parm->get_index());
7728 if (k != priv_->deleted_parms_by_id_.end())
7730 if (*k->second != *parm
7731 && (k->second->get_name_id() != parm_name))
7732 priv_->changed_parms_by_id_[parm->get_index()] =
7734 priv_->added_parms_.erase(parm_name);
7735 priv_->deleted_parms_.erase(k->second->get_name_id());
7736 priv_->deleted_parms_by_id_.erase(parm->get_index());
7739 priv_->added_parms_by_id_[parm->get_index()] = parm;
7745 priv_->sorted_subtype_changed_parms_);
7747 priv_->sorted_changed_parms_by_id_);
7749 priv_->sorted_deleted_parms_);
7752 priv_->sorted_added_parms_);
7762function_type_diff::deleted_parameter_at(
int i)
const
7768const vector<function_decl::parameter_sptr>&
7770{
return priv_->sorted_deleted_parms_;}
7775const vector<function_decl::parameter_sptr>&
7777{
return priv_->sorted_added_parms_;}
7786function_type_diff::inserted_parameter_at(
int i)
const
7814{
return dynamic_pointer_cast<function_type>(
first_subject());}
7830{
return priv_->return_type_diff_;}
7837{
return priv_->subtype_changed_parms_;}
7844{
return priv_->deleted_parms_;}
7851{
return priv_->added_parms_;}
7861 if (diff::priv_->pretty_representation_.empty())
7863 std::ostringstream o;
7864 o <<
"function_type_diff["
7869 diff::priv_->pretty_representation_ = o.str();
7871 return diff::priv_->pretty_representation_;
7895 return ir::NO_CHANGE_KIND;
7907 context()->get_reporter()->report(*
this, out, indent);
7921 for (vector<fn_parm_diff_sptr>::const_iterator i =
7922 priv_->sorted_subtype_changed_parms_.begin();
7923 i != priv_->sorted_subtype_changed_parms_.end();
7928 for (vector<fn_parm_diff_sptr>::const_iterator i =
7929 priv_->sorted_changed_parms_by_id_.begin();
7930 i != priv_->sorted_changed_parms_by_id_.end();
7953 if (!first || !second)
7962 first->get_parameters().end(),
7963 second->get_first_parm(),
7964 second->get_parameters().end(),
7965 result->priv_->parm_changes_);
7967 result->ensure_lookup_tables_populated();
7969 ctxt->initialize_canonical_diff(result);
7979function_decl_diff::ensure_lookup_tables_populated()
8016{
return dynamic_pointer_cast<function_decl>(
first_subject());}
8024function_decl_diff::type_diff()
const
8025{
return priv_->type_diff_;}
8032 if (diff::priv_->pretty_representation_.empty())
8034 std::ostringstream o;
8035 o <<
"function_diff["
8040 diff::priv_->pretty_representation_ = o.str();
8042 return diff::priv_->pretty_representation_;
8061 return ir::NO_CHANGE_KIND;
8073 context()->get_reporter()->report(*
this, out, indent);
8093 if (!first || !second)
8105 result->priv_->type_diff_ = type_diff;
8107 result->ensure_lookup_tables_populated();
8109 ctxt->initialize_canonical_diff(result);
8153 if (diff::priv_->pretty_representation_.empty())
8155 std::ostringstream o;
8156 o <<
"type_decl_diff["
8161 diff::priv_->pretty_representation_ = o.str();
8163 return diff::priv_->pretty_representation_;
8181 return ir::NO_CHANGE_KIND;
8192 context()->get_reporter()->report(*
this, out, indent);
8230 ctxt->initialize_canonical_diff(result);
8267 priv_(new
priv(underlying))
8275{
return dynamic_pointer_cast<typedef_decl>(
first_subject());}
8291{
return priv_->underlying_type_diff_;}
8300{priv_->underlying_type_diff_ = d;}
8307 if (diff::priv_->pretty_representation_.empty())
8309 std::ostringstream o;
8310 o <<
"typedef_diff["
8315 diff::priv_->pretty_representation_ = o.str();
8317 return diff::priv_->pretty_representation_;
8339 return ir::NO_CHANGE_KIND;
8351 context()->get_reporter()->report(*
this, out, indent);
8371 diff_sptr d = compute_diff_for_types(first->get_underlying_type(),
8372 second->get_underlying_type(),
8376 ctxt->initialize_canonical_diff(result);
8428 priv_(new
priv(first, second))
8437{
return priv_->first_;}
8444{
return priv_->second_;}
8458{
return ir::NO_CHANGE_KIND;}
8497 compute_diff(static_pointer_cast<scope_decl>(first->get_global_scope()),
8498 static_pointer_cast<scope_decl>(second->get_global_scope()),
8502 ctxt->initialize_canonical_diff(tu_diff);
8512struct diff_maps::priv
8535diff_maps::~diff_maps() =
default;
8542{
return priv_->type_decl_diff_map_;}
8549{
return priv_->type_decl_diff_map_;}
8556{
return priv_->enum_diff_map_;}
8563{
return priv_->enum_diff_map_;}
8570{
return priv_->class_diff_map_;}
8577{
return priv_->class_diff_map_;}
8584{
return priv_->union_diff_map_;}
8591{
return priv_->union_diff_map_;}
8598{
return priv_->typedef_diff_map_;}
8605{
return priv_->typedef_diff_map_;}
8612{
return priv_->subrange_diff_map_;}
8619{
return priv_->subrange_diff_map_;}
8626{
return priv_->array_diff_map_;}
8633{
return priv_->array_diff_map_;}
8640{
return priv_->reference_diff_map_;}
8647{{
return priv_->reference_diff_map_;}}
8654{
return priv_->fn_parm_diff_map_;}
8661{
return priv_->fn_parm_diff_map_;}
8668{
return priv_->function_type_diff_map_;}
8675{
return priv_->function_type_diff_map_;}
8682{
return priv_->function_decl_diff_map_;}
8689{
return priv_->function_decl_diff_map_;}
8696{
return priv_->var_decl_diff_map_;}
8703{
return priv_->var_decl_diff_map_;}
8710{
return priv_->distinct_diff_map_;}
8717{
return priv_->distinct_diff_map_;}
8775 diff_artifact_set_map_type::iterator i =
8776 priv_->impacted_artifacts_map_.find(dif);
8778 if (i == priv_->impacted_artifacts_map_.end())
8781 set.insert(impacted_iface);
8782 priv_->impacted_artifacts_map_[dif] = set;
8785 i->second.insert(impacted_iface);
8799 diff_artifact_set_map_type::iterator i =
8800 priv_->impacted_artifacts_map_.find(d);
8802 if (i == priv_->impacted_artifacts_map_.end())
8818 : priv_(new
priv(ctxt))
8826{
return priv_->num_func_removed;}
8833{priv_->num_func_removed = n;}
8843 if (priv_->ctxt() && !priv_->ctxt()->show_deleted_fns())
8844 return num_func_removed();
8845 return priv_->num_removed_func_filtered_out;
8854{priv_->num_removed_func_filtered_out = t;}
8865 ABG_ASSERT(num_func_removed() >= num_removed_func_filtered_out());
8866 return num_func_removed() - num_removed_func_filtered_out();
8874{
return priv_->num_func_added;}
8881{priv_->num_func_added = n;}
8889 if (priv_->ctxt() && !priv_->ctxt()->show_added_fns())
8890 return num_func_added();
8891 return priv_->num_added_func_filtered_out;
8900{priv_->num_added_func_filtered_out = n;}
8912 ABG_ASSERT(num_func_added() >= num_added_func_filtered_out());
8913 return num_func_added() - num_added_func_filtered_out();
8923{
return priv_->num_func_changed;}
8932{priv_->num_func_changed = n;}
8941{
return priv_->num_changed_func_filtered_out;}
8950{priv_->num_changed_func_filtered_out = n;}
8958{
return priv_->num_func_with_virt_offset_changes;}
8967{priv_->num_func_with_virt_offset_changes = n;}
8977{
return priv_->num_func_with_local_harmful_changes;}
8987{priv_->num_func_with_local_harmful_changes = n;}
8997{
return priv_->num_var_with_local_harmful_changes;}
9007{priv_->num_var_with_local_harmful_changes = n;}
9014{
return priv_->num_func_with_incompatible_changes;}
9021{priv_->num_func_with_incompatible_changes = n;}
9028{
return priv_->num_var_with_incompatible_changes;}
9035{priv_->num_var_with_incompatible_changes = n;}
9046{
return num_func_changed() - num_changed_func_filtered_out();}
9055{
return net_num_func_changed() - num_func_with_incompatible_changes();}
9062{
return priv_->num_vars_removed;}
9069{priv_->num_vars_removed = n;}
9078 if (priv_->ctxt() && !priv_->ctxt()->show_deleted_vars())
9079 return num_vars_removed();
9080 return priv_->num_removed_vars_filtered_out;
9089{priv_->num_removed_vars_filtered_out = n;}
9101 ABG_ASSERT(num_vars_removed() >= num_removed_vars_filtered_out());
9102 return num_vars_removed() - num_removed_vars_filtered_out();
9110{
return priv_->num_vars_added;}
9117{priv_->num_vars_added = n;}
9126 if (priv_->ctxt() && !priv_->ctxt()->show_added_vars())
9127 return num_vars_added();
9128 return priv_->num_added_vars_filtered_out;
9137{priv_->num_added_vars_filtered_out = n;}
9149 ABG_ASSERT(num_vars_added() >= num_added_vars_filtered_out());
9150 return num_vars_added() - num_added_vars_filtered_out();
9160{
return priv_->num_vars_changed;}
9169{priv_->num_vars_changed = n;}
9178{
return priv_->num_changed_vars_filtered_out;}
9187{priv_->num_changed_vars_filtered_out = n;}
9198{
return num_vars_changed() - num_changed_vars_filtered_out();}
9207{
return net_num_vars_changed() - num_var_with_incompatible_changes();}
9216{
return priv_->num_func_syms_removed;}
9225{priv_->num_func_syms_removed = n;}
9236 && !priv_->ctxt()->show_symbols_unreferenced_by_debug_info())
9237 return num_func_syms_removed();
9238 return priv_->num_removed_func_syms_filtered_out;
9248{priv_->num_removed_func_syms_filtered_out = n;}
9263 ABG_ASSERT(num_func_syms_removed() >= num_removed_func_syms_filtered_out());
9264 return num_func_syms_removed() - num_removed_func_syms_filtered_out();
9274{
return priv_->num_func_syms_added;}
9283{priv_->num_func_syms_added = n;}
9294 && !(priv_->ctxt()->show_added_symbols_unreferenced_by_debug_info()
9295 && priv_->ctxt()->show_symbols_unreferenced_by_debug_info()))
9296 return num_func_syms_added();
9297 return priv_->num_added_func_syms_filtered_out;
9307{priv_->num_added_func_syms_filtered_out = n;}
9322 ABG_ASSERT(num_func_syms_added() >= num_added_func_syms_filtered_out());
9323 return num_func_syms_added()- num_added_func_syms_filtered_out();
9333{
return priv_->num_var_syms_removed;}
9342{priv_->num_var_syms_removed = n;}
9353 && !priv_->ctxt()->show_symbols_unreferenced_by_debug_info())
9354 return num_var_syms_removed();
9355 return priv_->num_removed_var_syms_filtered_out;
9365{priv_->num_removed_var_syms_filtered_out = n;}
9380 ABG_ASSERT(num_var_syms_removed() >= num_removed_var_syms_filtered_out());
9381 return num_var_syms_removed() - num_removed_var_syms_filtered_out();
9391{
return priv_->num_var_syms_added;}
9400{priv_->num_var_syms_added = n;}
9411 && !(priv_->ctxt()->show_added_symbols_unreferenced_by_debug_info()
9412 && priv_->ctxt()->show_symbols_unreferenced_by_debug_info()))
9413 return num_var_syms_added();
9414 return priv_->num_added_var_syms_filtered_out;
9424{priv_->num_added_var_syms_filtered_out = n;}
9439 ABG_ASSERT(num_var_syms_added() >= num_added_var_syms_filtered_out());
9440 return num_var_syms_added() - num_added_var_syms_filtered_out();
9448{
return priv_->num_leaf_changes;}
9455{priv_->num_leaf_changes = n;}
9463{
return priv_->num_leaf_changes_filtered_out;}
9472{priv_->num_leaf_changes_filtered_out = n;}
9485 ABG_ASSERT(num_leaf_changes() >= num_leaf_changes_filtered_out());
9486 return num_leaf_changes() - num_leaf_changes_filtered_out();
9494{
return priv_->num_leaf_type_changes;}
9501{priv_->num_leaf_type_changes = n;}
9508{
return priv_->num_leaf_type_changes_filtered_out;}
9514{priv_->num_leaf_type_changes_filtered_out = n;}
9524{
return num_leaf_type_changes() - num_leaf_type_changes_filtered_out();}
9531{
return priv_->num_leaf_func_changes;}
9538{priv_->num_leaf_func_changes = n;}
9547{
return priv_->num_leaf_func_with_incompatible_changes;}
9556{priv_->num_leaf_func_with_incompatible_changes = n;}
9565{
return priv_->num_leaf_func_changes_filtered_out;}
9574{priv_->num_leaf_func_changes_filtered_out = n;}
9585{
return num_leaf_func_changes() - num_leaf_func_changes_filtered_out();}
9595 return (net_num_leaf_func_changes()
9596 - num_leaf_func_with_incompatible_changes());
9604{
return priv_->num_leaf_var_changes;}
9611{priv_->num_leaf_var_changes = n;}
9620{
return priv_->num_leaf_var_with_incompatible_changes;}
9629{priv_->num_leaf_var_with_incompatible_changes = n;}
9641{
return priv_->num_added_unreachable_types;}
9654{priv_->num_added_unreachable_types = n;}
9665{
return priv_->num_added_unreachable_types_filtered_out;}
9676{priv_->num_added_unreachable_types_filtered_out = n;}
9690 num_added_unreachable_types_filtered_out());
9692 return (num_added_unreachable_types()
9694 num_added_unreachable_types_filtered_out());
9707{
return priv_->num_removed_unreachable_types;}
9719{priv_->num_removed_unreachable_types = n;}
9730{
return priv_->num_removed_unreachable_types_filtered_out;}
9741{priv_->num_removed_unreachable_types_filtered_out = n;}
9755 num_removed_unreachable_types_filtered_out());
9757 return (num_removed_unreachable_types()
9759 num_removed_unreachable_types_filtered_out());
9772{
return priv_->num_changed_unreachable_types;}
9784{priv_->num_changed_unreachable_types = n;}
9795{
return priv_->num_changed_unreachable_types_filtered_out;}
9806{priv_->num_changed_unreachable_types_filtered_out = n;}
9820 num_changed_unreachable_types_filtered_out());
9822 return (num_changed_unreachable_types()
9824 num_changed_unreachable_types_filtered_out());
9834{
return priv_->num_leaf_var_changes_filtered_out;}
9843{priv_->num_leaf_var_changes_filtered_out = n;}
9854{
return num_leaf_var_changes() - num_leaf_var_changes_filtered_out();}
9863{
return net_num_leaf_var_changes() - num_leaf_var_with_incompatible_changes();}
9871{
return ctxt_.lock();}
9879 return (deleted_fns_.empty()
9880 && added_fns_.empty()
9881 && changed_fns_map_.empty()
9882 && deleted_vars_.empty()
9883 && added_vars_.empty()
9884 && changed_vars_map_.empty());
9891 deleted_fns_.clear();
9893 changed_fns_map_.clear();
9894 deleted_vars_.clear();
9895 added_vars_.clear();
9896 changed_vars_map_.clear();
9904 if (!lookup_tables_empty())
9912 for (vector<deletion>::const_iterator it = e.deletions().begin();
9913 it != e.deletions().end();
9916 unsigned i = it->index();
9917 ABG_ASSERT(i < first_->get_functions().size());
9919 const function_decl* deleted_fn = first_->get_functions()[i];
9926 deleted_fns_[n] = deleted_fn;
9929 for (vector<insertion>::const_iterator it = e.insertions().begin();
9930 it != e.insertions().end();
9933 for (vector<unsigned>::const_iterator iit =
9934 it->inserted_indexes().begin();
9935 iit != it->inserted_indexes().end();
9939 const function_decl* added_fn = second_->get_functions()[i];
9946 string_function_ptr_map::const_iterator j =
9947 deleted_fns_.find(n);
9948 if (j != deleted_fns_.end())
9955 if (*j->second != *added_fn)
9956 changed_fns_map_[j->first] = d;
9957 deleted_fns_.erase(j);
9960 added_fns_[n] = added_fn;
9969 vector<string> to_delete;
9970 for (string_function_ptr_map::const_iterator i = deleted_fns_.begin();
9971 i != deleted_fns_.end();
9973 if (second_->lookup_function_symbol(*i->second->get_symbol()))
9974 to_delete.push_back(i->first);
9976 for (vector<string>::const_iterator i = to_delete.begin();
9977 i != to_delete.end();
9979 deleted_fns_.erase(*i);
9984 for (string_function_ptr_map::const_iterator i = added_fns_.begin();
9985 i != added_fns_.end();
9988 if (first_->lookup_function_symbol(*i->second->get_symbol()))
9989 to_delete.push_back(i->first);
9990 else if (! i->second->get_symbol()->get_version().is_empty()
9991 && i->second->get_symbol()->get_version().is_default())
9999 if (first_->lookup_function_symbol(i->second->get_symbol()->get_name(),
10001 to_delete.push_back(i->first);
10005 for (vector<string>::const_iterator i = to_delete.begin();
10006 i != to_delete.end();
10008 added_fns_.erase(*i);
10014 for (vector<deletion>::const_iterator it = e.deletions().begin();
10015 it != e.deletions().end();
10018 unsigned i = it->index();
10019 ABG_ASSERT(i < first_->get_variables().size());
10021 const var_decl_sptr deleted_var = first_->get_variables()[i];
10022 string n = deleted_var->get_id();
10039 string_var_ptr_map::const_iterator j = deleted_vars_.find(n);
10040 if (j != deleted_vars_.end())
10047 deleted_vars_[n] = deleted_var;
10050 for (vector<insertion>::const_iterator it = e.insertions().begin();
10051 it != e.insertions().end();
10054 for (vector<unsigned>::const_iterator iit =
10055 it->inserted_indexes().begin();
10056 iit != it->inserted_indexes().end();
10060 const var_decl_sptr added_var = second_->get_variables()[i];
10061 string n = added_var->get_id();
10064 string_var_ptr_map::const_iterator k = added_vars_.find(n);
10065 if ( k != added_vars_.end())
10072 string_var_ptr_map::const_iterator j =
10073 deleted_vars_.find(n);
10074 if (j != deleted_vars_.end())
10076 if (*j->second != *added_var)
10082 deleted_vars_.erase(j);
10085 added_vars_[n] = added_var;
10089 sorted_changed_vars_);
10095 vector<string> to_delete;
10096 for (string_var_ptr_map::const_iterator i = deleted_vars_.begin();
10097 i != deleted_vars_.end();
10099 if (second_->lookup_variable_symbol(*i->second->get_symbol()))
10100 to_delete.push_back(i->first);
10102 for (vector<string>::const_iterator i = to_delete.begin();
10103 i != to_delete.end();
10105 deleted_vars_.erase(*i);
10110 for (string_var_ptr_map::const_iterator i = added_vars_.begin();
10111 i != added_vars_.end();
10113 if (first_->lookup_variable_symbol(*i->second->get_symbol()))
10114 to_delete.push_back(i->first);
10115 else if (! i->second->get_symbol()->get_version().is_empty()
10116 && i->second->get_symbol()->get_version().is_default())
10124 if (first_->lookup_variable_symbol(i->second->get_symbol()->get_name(),
10126 to_delete.push_back(i->first);
10129 for (vector<string>::const_iterator i = to_delete.begin();
10130 i != to_delete.end();
10132 added_vars_.erase(*i);
10140 for (vector<deletion>::const_iterator it = e.deletions().begin();
10141 it != e.deletions().end();
10144 unsigned i = it->index();
10145 ABG_ASSERT(i < first_->get_unreferenced_function_symbols().size());
10147 first_->get_unreferenced_function_symbols()[i];
10148 if (!second_->lookup_function_symbol(*deleted_sym))
10149 deleted_unrefed_fn_syms_[deleted_sym->get_id_string()] = deleted_sym;
10152 for (vector<insertion>::const_iterator it = e.insertions().begin();
10153 it != e.insertions().end();
10156 for (vector<unsigned>::const_iterator iit =
10157 it->inserted_indexes().begin();
10158 iit != it->inserted_indexes().end();
10162 ABG_ASSERT(i < second_->get_unreferenced_function_symbols().size());
10164 second_->get_unreferenced_function_symbols()[i];
10165 if ((deleted_unrefed_fn_syms_.find(added_sym->get_id_string())
10166 == deleted_unrefed_fn_syms_.end()))
10168 if (!first_->lookup_function_symbol(*added_sym))
10170 bool do_add =
true;
10171 if (! added_sym->get_version().is_empty()
10172 && added_sym->get_version().is_default())
10180 if (first_->lookup_function_symbol(added_sym->get_name(),
10186 added_unrefed_fn_syms_[added_sym->get_id_string()] =
10191 deleted_unrefed_fn_syms_.erase(added_sym->get_id_string());
10201 for (vector<deletion>::const_iterator it = e.deletions().begin();
10202 it != e.deletions().end();
10205 unsigned i = it->index();
10206 ABG_ASSERT(i < first_->get_unreferenced_variable_symbols().size());
10208 first_->get_unreferenced_variable_symbols()[i];
10209 if (!second_->lookup_variable_symbol(*deleted_sym))
10210 deleted_unrefed_var_syms_[deleted_sym->get_id_string()] = deleted_sym;
10213 for (vector<insertion>::const_iterator it = e.insertions().begin();
10214 it != e.insertions().end();
10217 for (vector<unsigned>::const_iterator iit =
10218 it->inserted_indexes().begin();
10219 iit != it->inserted_indexes().end();
10223 ABG_ASSERT(i < second_->get_unreferenced_variable_symbols().size());
10225 second_->get_unreferenced_variable_symbols()[i];
10226 if (deleted_unrefed_var_syms_.find(added_sym->get_id_string())
10227 == deleted_unrefed_var_syms_.end())
10229 if (!first_->lookup_variable_symbol(*added_sym))
10231 bool do_add =
true;
10232 if (! added_sym->get_version().is_empty()
10233 && added_sym->get_version().is_default())
10241 if (first_->lookup_variable_symbol(added_sym->get_name(),
10247 added_unrefed_var_syms_[added_sym->get_id_string()] =
10252 deleted_unrefed_var_syms_.erase(added_sym->get_id_string());
10263 for (vector<deletion>::const_iterator it = e.deletions().begin();
10264 it != e.deletions().end();
10267 unsigned i = it->index();
10269 (first_->get_types_not_reachable_from_public_interfaces()[i]);
10276 deleted_unreachable_types_[repr] = t;
10281 for (vector<insertion>::const_iterator it = e.insertions().begin();
10282 it != e.insertions().end();
10285 for (vector<unsigned>::const_iterator iit =
10286 it->inserted_indexes().begin();
10287 iit != it->inserted_indexes().end();
10292 (second_->get_types_not_reachable_from_public_interfaces()[i]);
10311 string_type_base_sptr_map::const_iterator j =
10312 deleted_unreachable_types_.find(repr);
10313 if (j != deleted_unreachable_types_.end())
10318 decl_base_sptr old_type =
is_decl(j->second);
10319 decl_base_sptr new_type =
is_decl(t);
10320 if (old_type != new_type)
10328 changed_unreachable_types_[repr]= d;
10334 deleted_unreachable_types_.erase(j);
10339 added_unreachable_types_[repr] = t;
10352 std::set<type_base_sptr> deleted_anon_types;
10353 std::set<type_base_sptr> added_anon_types;
10355 for (
auto entry : deleted_unreachable_types_)
10361 deleted_anon_types.insert(entry.second);
10365 for (
auto entry : added_unreachable_types_)
10370 added_anon_types.insert(entry.second);
10375 class_or_union_sptr deleted_class;
10380 for (
auto deleted: deleted_anon_types)
10389 for (
auto enr : deleted_enum->get_enumerators())
10391 bool this_enum_got_changed =
false;
10392 for (
auto t : added_anon_types)
10409 changed_unreachable_types_[repr]= d;
10410 this_enum_got_changed =
true;
10417 removed_anon_types_to_erase[r1] = deleted_enum;
10418 added_anon_types_to_erase[r2] = added_enum;
10422 if (this_enum_got_changed)
10426 else if (deleted_class)
10431 for (
auto dm : deleted_class->get_data_members())
10433 bool this_class_got_changed =
false;
10434 for (
auto klass : added_anon_types)
10436 if (class_or_union_sptr added_class =
10455 changed_unreachable_types_[repr]= d;
10456 this_class_got_changed =
true;
10463 removed_anon_types_to_erase[r1] = deleted_class;
10464 added_anon_types_to_erase[r2] = added_class;
10468 if (this_class_got_changed)
10477 for (
auto entry : added_anon_types_to_erase)
10478 added_unreachable_types_.erase(entry.first);
10480 for (
auto entry : removed_anon_types_to_erase)
10481 deleted_unreachable_types_.erase(entry.first);
10510 return fn_suppr->suppresses_function(fn, k, ctxt);
10537 return var_suppr->suppresses_variable(var, k, ctxt);
10549 for (suppressions_type::const_iterator i = suppressions.begin();
10550 i != suppressions.end();
10557 for (string_function_ptr_map::const_iterator e = added_fns_.begin();
10558 e != added_fns_.end();
10560 if (function_is_suppressed(e->second, fn_suppr,
10563 suppressed_added_fns_[e->first] = e->second;
10566 for (string_function_ptr_map::const_iterator e = deleted_fns_.begin();
10567 e != deleted_fns_.end();
10569 if (function_is_suppressed(e->second, fn_suppr,
10572 suppressed_deleted_fns_[e->first] = e->second;
10575 for (string_elf_symbol_map::const_iterator e =
10576 added_unrefed_fn_syms_.begin();
10577 e != added_unrefed_fn_syms_.end();
10579 if (fn_suppr->suppresses_function_symbol(e->second,
10582 suppressed_added_unrefed_fn_syms_[e->first] = e->second;
10585 for (string_elf_symbol_map::const_iterator e =
10586 deleted_unrefed_fn_syms_.begin();
10587 e != deleted_unrefed_fn_syms_.end();
10589 if (fn_suppr->suppresses_function_symbol(e->second,
10592 suppressed_deleted_unrefed_fn_syms_[e->first] = e->second;
10600 for (string_function_ptr_map::const_iterator e = added_fns_.begin();
10601 e != added_fns_.end();
10610 if (type_suppr->suppresses_type(c, ctxt))
10611 suppressed_added_fns_[e->first] = e->second;
10614 for (string_function_ptr_map::const_iterator e = deleted_fns_.begin();
10615 e != deleted_fns_.end();
10624 if (type_suppr->suppresses_type(c, ctxt))
10625 suppressed_deleted_fns_[e->first] = e->second;
10630 for (string_type_base_sptr_map::const_iterator e =
10631 deleted_unreachable_types_.begin();
10632 e != deleted_unreachable_types_.end();
10634 if (type_suppr->suppresses_type(e->second, ctxt))
10635 suppressed_deleted_unreachable_types_[e->first] = e->second;
10639 for (string_type_base_sptr_map::const_iterator e =
10640 added_unreachable_types_.begin();
10641 e != added_unreachable_types_.end();
10643 if (type_suppr->suppresses_type(e->second, ctxt))
10644 suppressed_added_unreachable_types_[e->first] = e->second;
10651 for (string_var_ptr_map::const_iterator e = added_vars_.begin();
10652 e != added_vars_.end();
10654 if (variable_is_suppressed(e->second, var_suppr,
10657 suppressed_added_vars_[e->first] = e->second;
10660 for (string_var_ptr_map::const_iterator e = deleted_vars_.begin();
10661 e != deleted_vars_.end();
10663 if (variable_is_suppressed(e->second, var_suppr,
10666 suppressed_deleted_vars_[e->first] = e->second;
10669 for (string_elf_symbol_map::const_iterator e =
10670 added_unrefed_var_syms_.begin();
10671 e != added_unrefed_var_syms_.end();
10673 if (var_suppr->suppresses_variable_symbol(e->second,
10676 suppressed_added_unrefed_var_syms_[e->first] = e->second;
10679 for (string_elf_symbol_map::const_iterator e =
10680 deleted_unrefed_var_syms_.begin();
10681 e != deleted_unrefed_var_syms_.end();
10683 if (var_suppr->suppresses_variable_symbol(e->second,
10686 suppressed_deleted_unrefed_var_syms_[e->first] = e->second;
10704 string_function_ptr_map::const_iterator i =
10705 suppressed_deleted_fns_.find(fn->
get_id());
10707 return (i != suppressed_deleted_fns_.end());
10724 string_type_base_sptr_map::const_iterator i =
10725 suppressed_added_unreachable_types_.find(repr);
10726 if (i == suppressed_added_unreachable_types_.end())
10746 string_type_base_sptr_map::const_iterator i =
10747 suppressed_deleted_unreachable_types_.find(repr);
10748 if (i == suppressed_deleted_unreachable_types_.end())
10767 string_function_ptr_map::const_iterator i =
10768 suppressed_added_fns_.find(fn->
get_id());
10770 return (i != suppressed_added_fns_.end());
10786 string_var_ptr_map::const_iterator i =
10787 suppressed_deleted_vars_.find(var->get_id());
10789 return (i != suppressed_deleted_vars_.end());
10805 string_var_ptr_map::const_iterator i =
10806 suppressed_added_vars_.find(var->get_id());
10808 return (i != suppressed_added_vars_.end());
10824 string_elf_symbol_map::const_iterator i =
10825 suppressed_deleted_unrefed_fn_syms_.find(s->
get_id_string());
10827 return (i != suppressed_deleted_unrefed_fn_syms_.end());
10843 string_elf_symbol_map::const_iterator i =
10844 suppressed_added_unrefed_fn_syms_.find(s->
get_id_string());
10846 return (i != suppressed_added_unrefed_fn_syms_.end());
10862 string_elf_symbol_map::const_iterator i =
10863 suppressed_deleted_unrefed_var_syms_.find(s->
get_id_string());
10865 return (i != suppressed_deleted_unrefed_var_syms_.end());
10881 string_elf_symbol_map::const_iterator i =
10882 suppressed_added_unrefed_var_syms_.find(s->
get_id_string());
10884 return (i != suppressed_added_unrefed_var_syms_.end());
10887#ifdef do_count_diff_map_changes
10888#undef do_count_diff_map_changes
10890#define do_count_diff_map_changes(diff_map, n_changes, n_filtered) \
10892 string_diff_ptr_map::const_iterator i; \
10893 for (i = diff_map.begin(); \
10894 i != diff_map.end(); \
10897 if (const var_diff* d = is_var_diff(i->second)) \
10898 if (is_data_member(d->first_var())) \
10901 if (i->second->has_local_changes()) \
10903 if (!i->second->get_canonical_diff()->to_be_reported()) \
10919 count_leaf_type_changes(num_changes, num_filtered);
10922 do_count_diff_map_changes(leaf_diffs_.get_function_decl_diff_map(),
10923 num_changes, num_filtered);
10924 do_count_diff_map_changes(leaf_diffs_.get_var_decl_diff_map(),
10925 num_changes, num_filtered);
10938 size_t &num_filtered)
10940 do_count_diff_map_changes(leaf_diffs_.get_type_decl_diff_map(),
10941 num_changes, num_filtered);
10942 do_count_diff_map_changes(leaf_diffs_.get_enum_diff_map(),
10943 num_changes, num_filtered);
10944 do_count_diff_map_changes(leaf_diffs_.get_class_diff_map(),
10945 num_changes, num_filtered);
10946 do_count_diff_map_changes(leaf_diffs_.get_union_diff_map(),
10947 num_changes, num_filtered);
10948 do_count_diff_map_changes(leaf_diffs_.get_typedef_diff_map(),
10949 num_changes, num_filtered);
10950 do_count_diff_map_changes(leaf_diffs_.get_subrange_diff_map(),
10951 num_changes, num_filtered);
10952 do_count_diff_map_changes(leaf_diffs_.get_array_diff_map(),
10953 num_changes, num_filtered);
10954 do_count_diff_map_changes(leaf_diffs_.get_distinct_diff_map(),
10955 num_changes, num_filtered);
10956 do_count_diff_map_changes(leaf_diffs_.get_fn_parm_diff_map(),
10957 num_changes, num_filtered);
10985 size_t &num_deleted,
10986 size_t &num_changed,
10987 size_t &num_filtered_added,
10988 size_t &num_filtered_deleted,
10989 size_t &num_filtered_changed)
10991 num_added = added_unreachable_types_.size();
10992 num_deleted = deleted_unreachable_types_.size();
10993 num_changed = changed_unreachable_types_.size();
10994 num_filtered_added = suppressed_added_unreachable_types_.size();
10995 num_filtered_deleted = suppressed_deleted_unreachable_types_.size();
10997 for (vector<diff_sptr>::const_iterator i =
11001 if (!(*i)->to_be_reported())
11002 ++num_filtered_changed;
11011{
return changed_unreachable_types_;}
11022const vector<diff_sptr>&
11025if (changed_unreachable_types_sorted_.empty())
11026 if (!changed_unreachable_types_.empty())
11028 changed_unreachable_types_sorted_);
11030 return changed_unreachable_types_sorted_;
11065 if (ctxt->perform_change_categorization())
11067 if (get_context()->
do_log())
11069 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11070 <<
"applying filters to "
11071 << changed_fns_.size()
11072 <<
" changed fns ...\n";
11078 for (function_decl_diff_sptrs_type::const_iterator i =
11079 changed_fns_.begin();
11080 i != changed_fns_.end();
11084 ctxt->maybe_apply_filters(
diff);
11087 if (get_context()->
do_log())
11090 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11091 <<
"filters to changed fn applied!:" << t <<
"\n";
11093 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11094 <<
"applying filters to "
11095 << sorted_changed_vars_.size()
11096 <<
" changed vars ...\n";
11102 for (var_diff_sptrs_type::const_iterator i = sorted_changed_vars_.begin();
11103 i != sorted_changed_vars_.end();
11107 ctxt->maybe_apply_filters(
diff);
11110 if (get_context()->
do_log())
11113 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11114 <<
"filters to changed vars applied!:" << t <<
"\n";
11116 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11117 <<
"applying filters to unreachable types ...\n";
11124 ctxt->maybe_apply_filters(
diff);
11127 ctxt->maybe_apply_filters(entry.second);
11129 if (get_context()->
do_log())
11132 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11133 <<
"filters to unreachable types applied!:" << t <<
"\n";
11135 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11136 <<
"categorizing redundant changed sub nodes ...\n";
11140 categorize_redundant_changed_sub_nodes();
11142 if (get_context()->
do_log())
11145 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11146 <<
"redundant changed sub nodes categorized!:" << t <<
"\n";
11148 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11149 <<
"count changed fns ...\n";
11157 for (function_decl_diff_sptrs_type::const_iterator i =
11158 changed_fns_.begin();
11159 i != changed_fns_.end();
11162 bool incompatible_change =
false;
11163 if ((*i)->is_filtered_out())
11168 if ((*i)->has_local_changes())
11174 if (!(*i)->is_categorized_as_suppressed())
11188 incompatible_change =
true;
11198 incompatible_change =
true;
11202 if ((*i)->has_local_changes())
11206 if (incompatible_change)
11208 incompatible_changed_fns_.push_back(*i);
11212 if ((*i)->has_local_changes())
11217 if ((*i)->is_filtered_out())
11225 if ((*i)->has_local_changes())
11232 if (get_context()->
do_log())
11235 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11236 <<
"changed fn counted!:" << t <<
"\n";
11238 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11239 <<
"count changed vars ...\n";
11246 for (var_diff_sptrs_type ::const_iterator i = sorted_changed_vars_.begin();
11247 i != sorted_changed_vars_.end();
11250 bool incompatible_change =
false;
11251 if ((*i)->is_filtered_out())
11256 if ((*i)->has_local_changes())
11261 if (!(*i)->is_categorized_as_suppressed())
11265 incompatible_change =
true;
11266 incompatible_changed_vars_.push_back(*i);
11272 if ((*i)->is_filtered_out())
11277 if ((*i)->has_local_changes())
11284 if ((*i)->has_local_changes())
11289 if (incompatible_change)
11295 if (get_context()->
do_log())
11298 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11299 <<
"changed vars counted!:" << t <<
"\n";
11301 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11302 <<
"count leaf changed types ...\n";
11317 size_t num_type_changes = 0, num_type_filtered = 0;
11318 count_leaf_type_changes(num_type_changes, num_type_filtered);
11324 if (get_context()->
do_log())
11327 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11328 <<
"changed leaf types counted!:" << t <<
"\n";
11330 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11331 <<
"count leaf changed artefacts ...\n";
11337 size_t num_changes = 0, num_filtered = 0;
11338 count_leaf_changes(num_changes, num_filtered);
11344 if (get_context()->
do_log())
11347 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11348 <<
"changed leaf artefacts counted!:" << t <<
"\n";
11350 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11351 <<
"count unreachable types ...\n";
11357 size_t num_added_unreachable_types = 0,
11358 num_changed_unreachable_types = 0,
11359 num_deleted_unreachable_types = 0,
11360 num_added_unreachable_types_filtered = 0,
11361 num_changed_unreachable_types_filtered = 0,
11362 num_deleted_unreachable_types_filtered = 0;
11364 count_unreachable_types(num_added_unreachable_types,
11365 num_deleted_unreachable_types,
11366 num_changed_unreachable_types,
11367 num_added_unreachable_types_filtered,
11368 num_deleted_unreachable_types_filtered,
11369 num_changed_unreachable_types_filtered);
11371 if (get_context()->
do_log())
11374 std::cerr <<
"in apply_filters_and_compute_diff_stats:"
11375 <<
"unreachable types counted!:" << t <<
"\n";
11382 (num_added_unreachable_types_filtered);
11384 (num_deleted_unreachable_types_filtered);
11386 (num_changed_unreachable_types_filtered);
11402 const string& indent)
11405 size_t net_num_leaf_changes =
11418 if (!sonames_equal_)
11419 out << indent <<
"ELF SONAME changed\n";
11421 if (!architectures_equal_)
11422 out << indent <<
"ELF architecture changed\n";
11426 if (ctxt->show_leaf_changes_only())
11428 out <<
"Leaf changes summary: ";
11429 out << net_num_leaf_changes <<
" artifact";
11430 if (net_num_leaf_changes > 1)
11435 out <<
" (" << num_filtered <<
" filtered out)";
11438 out << indent <<
"Changed leaf types summary: "
11442 <<
" filtered out)";
11443 out <<
" leaf type";
11446 out <<
" changed\n";
11449 out << indent <<
"Removed/Changed/Added functions summary: ";
11454 <<
" filtered out)";
11461 <<
" filtered out)";
11468 out <<
"functions";
11474 out << indent <<
"Removed/Changed/Added variables summary: ";
11478 <<
" filtered out)";
11485 <<
" filtered out)";
11492 out <<
"variables";
11495 <<
" filtered out)";
11504 out << indent <<
"Functions changes summary: ";
11509 <<
" filtered out)";
11520 if (total_nb_function_changes <= 1)
11521 out <<
" function";
11523 out <<
" functions";
11530 out << indent <<
"Variables changes summary: ";
11534 <<
" filtered out)";
11545 <<
" filtered out)";
11546 if (total_nb_variable_changes <= 1)
11547 out <<
" variable";
11549 out <<
" variables";
11555 if (ctxt->show_unreachable_types())
11557 size_t total_nb_unreachable_type_changes =
11563 out << indent <<
"Unreachable types summary: "
11568 <<
" filtered out)";
11575 <<
" filtered out)";
11582 <<
" filtered out)";
11583 if (total_nb_unreachable_type_changes <= 1)
11590 if (ctxt->show_symbols_unreferenced_by_debug_info()
11598 if (!ctxt->show_added_symbols_unreferenced_by_debug_info()
11608 <<
"Function symbols changes summary: "
11612 <<
" filtered out)";
11617 <<
" filtered out)";
11618 out <<
" function symbol";
11621 out <<
" not referenced by debug info\n";
11626 if (!ctxt->show_added_symbols_unreferenced_by_debug_info()
11636 <<
"Variable symbols changes summary: "
11640 <<
" filtered out)";
11645 <<
" filtered out)";
11646 out <<
" variable symbol";
11649 out <<
" not referenced by debug info\n";
11663 ctxt->forget_visited_diffs();
11664 for (function_decl_diff_sptrs_type::const_iterator i =
11665 changed_fns_.begin();
11666 i!= changed_fns_.end();
11673 for (var_diff_sptrs_type::const_iterator i = sorted_changed_vars_.begin();
11674 i!= sorted_changed_vars_.end();
11681 for (diff_sptrs_type::const_iterator i =
11697 for (function_decl_diff_sptrs_type::const_iterator i = changed_fns_.begin();
11698 i!= changed_fns_.end();
11705 for (var_diff_sptrs_type::const_iterator i = sorted_changed_vars_.begin();
11706 i!= sorted_changed_vars_.end();
11724 if (!ctxt->dump_diff_tree()
11725 || ctxt->error_output_stream() == 0)
11728 if (!changed_fns_.empty())
11730 *ctxt->error_output_stream() <<
"changed functions diff tree: \n\n";
11731 for (function_decl_diff_sptrs_type::const_iterator i =
11732 changed_fns_.begin();
11733 i != changed_fns_.end();
11741 if (!sorted_changed_vars_.empty())
11743 *ctxt->error_output_stream() <<
"\nchanged variables diff tree: \n\n";
11744 for (var_diff_sptrs_type::const_iterator i =
11745 sorted_changed_vars_.begin();
11746 i != sorted_changed_vars_.end();
11756 *ctxt->error_output_stream() <<
"\nchanged unreachable "
11757 "types diff tree: \n\n";
11758 for (vector<diff_sptr>::const_iterator i =
11776 for (function_decl_diff_sptrs_type::const_iterator i =
11795 corpus_sptr second,
11797 : priv_(new
priv(first, second, ctxt))
11800corpus_diff::~corpus_diff() =
default;
11806 if (priv_->finished_)
11809 priv_->finished_ =
true;
11829{
return priv_->first_;}
11834{
return priv_->second_;}
11837const vector<diff*>&
11839{
return priv_->children_;}
11857 bool inserted =
false;
11858 for (vector<diff*>::iterator i = priv_->children_.begin();
11859 i != priv_->children_.end();
11864 context()->keep_diff_alive(d);
11865 priv_->children_.insert(i, d.get());
11874 context()->keep_diff_alive(d);
11878 priv_->children_.push_back(d.get());
11886{
return priv_->fns_edit_script_;}
11892{
return priv_->vars_edit_script_;}
11899{
return !priv_->sonames_equal_;}
11906{
return !priv_->architectures_equal_;}
11913{
return priv_->deleted_fns_;}
11920{
return priv_->added_fns_;}
11932{
return priv_->changed_fns_map_;}
11941{
return priv_->changed_fns_;}
11950{
return priv_->incompatible_changed_fns_;}
11959{
return priv_->incompatible_changed_fns_;}
11967{
return priv_->deleted_vars_;}
11974{
return priv_->added_vars_;}
11982{
return priv_->changed_vars_map_;}
11990{
return priv_->sorted_changed_vars_;}
11999{
return priv_->incompatible_changed_vars_;}
12008{
return priv_->incompatible_changed_vars_;}
12017{
return priv_->deleted_unrefed_fn_syms_;}
12026{
return priv_->added_unrefed_fn_syms_;}
12035{
return priv_->deleted_unrefed_var_syms_;}
12044{
return priv_->added_unrefed_var_syms_;}
12053{
return priv_->deleted_unreachable_types_;}
12061const vector<type_base_sptr>&
12064 if (priv_->deleted_unreachable_types_sorted_.empty())
12065 if (!priv_->deleted_unreachable_types_.empty())
12067 priv_->deleted_unreachable_types_sorted_);
12069 return priv_->deleted_unreachable_types_sorted_;
12079{
return priv_->added_unreachable_types_;}
12087const vector<type_base_sptr>&
12090 if (priv_->added_unreachable_types_sorted_.empty())
12091 if (!priv_->added_unreachable_types_.empty())
12093 priv_->added_unreachable_types_sorted_);
12095 return priv_->added_unreachable_types_sorted_;
12105{
return priv_->changed_unreachable_types_;}
12113const vector<diff_sptr>&
12115{
return priv_->changed_unreachable_types_sorted();}
12122{
return priv_->get_context();}
12129 if (priv_->pretty_representation_.empty())
12131 std::ostringstream o;
12132 o <<
"corpus_diff["
12137 priv_->pretty_representation_ = o.str();
12139 return priv_->pretty_representation_;
12150 || !(priv_->deleted_fns_.empty()
12151 && priv_->added_fns_.empty()
12152 && priv_->changed_fns_map_.empty()
12153 && priv_->deleted_vars_.empty()
12154 && priv_->added_vars_.empty()
12155 && priv_->changed_vars_map_.empty()
12156 && priv_->added_unrefed_fn_syms_.empty()
12157 && priv_->deleted_unrefed_fn_syms_.empty()
12158 && priv_->added_unrefed_var_syms_.empty()
12159 && priv_->deleted_unrefed_var_syms_.empty()
12160 && priv_->deleted_unreachable_types_.empty()
12161 && priv_->added_unreachable_types_.empty()
12162 && priv_->changed_unreachable_types_.empty()));
12220 for (
auto &entry : priv_->changed_unreachable_types())
12264{
return context()->get_reporter()->diff_has_net_changes(
this);}
12289 if (priv_->diff_stats_)
12290 return *priv_->diff_stats_;
12295 std::cerr <<
"Applying suppressions ...\n";
12304 std::cerr <<
"suppressions applied!:" << t <<
"\n";
12311 std::cerr <<
"Marking leaf nodes ...\n";
12320 std::cerr <<
"leaf nodes marked!:" << t <<
"\n";
12321 std::cerr <<
"Applying filters and computing diff stats ...\n";
12325 priv_->apply_filters_and_compute_diff_stats(*priv_->diff_stats_);
12330 std::cerr <<
"Filters applied and diff stats computed!: " << t <<
"\n";
12333 return *priv_->diff_stats_;
12355 visit_begin(
diff *d)
12402 const corpus_diff *corpus_diff_node = ctxt->get_corpus_diff().get();
12405 if (
diff *iface_diff = get_current_topmost_iface_diff())
12412 get_leaf_diffs().insert_diff_node(d, iface);
12430 if (!
context()->show_leaf_changes_only())
12433 leaf_diff_node_marker_visitor v;
12434 context()->forget_visited_diffs();
12435 bool s =
context()->visiting_a_node_twice_is_forbidden();
12436 context()->forbid_visiting_a_node_twice(
true);
12437 if (
context()->show_impacted_interfaces())
12438 context()->forbid_visiting_a_node_twice_per_interface(
true);
12440 context()->forbid_visiting_a_node_twice(s);
12441 context()->forbid_visiting_a_node_twice_per_interface(
false);
12451{
return priv_->leaf_diffs_;}
12460{
return priv_->leaf_diffs_;}
12471 context()->get_reporter()->report(*
this, out, indent);
12486 if (!v.
visit(
this,
true))
12492 for (function_decl_diff_sptrs_type::const_iterator i =
12500 if (ctxt->visiting_a_node_twice_is_forbidden_per_interface())
12501 ctxt->forget_visited_diffs();
12514 for (var_diff_sptrs_type::const_iterator i =
12522 if (ctxt->visiting_a_node_twice_is_forbidden_per_interface())
12523 ctxt->forget_visited_diffs();
12540 for (vector<diff_sptr>::const_iterator i =
12548 if (ctxt->visiting_a_node_twice_is_forbidden_per_interface())
12549 ctxt->forget_visited_diffs();
12577 const corpus_sptr s,
12580 typedef corpus::functions::const_iterator fns_it_type;
12581 typedef corpus::variables::const_iterator vars_it_type;
12582 typedef elf_symbols::const_iterator symbols_it_type;
12584 typedef vector<type_base_wptr>::const_iterator type_base_wptr_it_type;
12593 ctxt->set_corpus_diff(r);
12595 if(ctxt->show_soname_change())
12596 r->priv_->sonames_equal_ = f->get_soname() == s->get_soname();
12598 r->priv_->sonames_equal_ =
true;
12600 r->priv_->architectures_equal_ =
12601 f->get_architecture_name() == s->get_architecture_name();
12604 diff_utils::compute_diff<fns_it_type, eq_type>(f->get_functions().begin(),
12605 f->get_functions().end(),
12606 s->get_functions().begin(),
12607 s->get_functions().end(),
12608 r->priv_->fns_edit_script_);
12611 diff_utils::compute_diff<vars_it_type, eq_type>
12612 (f->get_variables().begin(), f->get_variables().end(),
12613 s->get_variables().begin(), s->get_variables().end(),
12614 r->priv_->vars_edit_script_);
12618 diff_utils::compute_diff<symbols_it_type, eq_type>
12619 (f->get_unreferenced_function_symbols().begin(),
12620 f->get_unreferenced_function_symbols().end(),
12621 s->get_unreferenced_function_symbols().begin(),
12622 s->get_unreferenced_function_symbols().end(),
12623 r->priv_->unrefed_fn_syms_edit_script_);
12627 diff_utils::compute_diff<symbols_it_type, eq_type>
12628 (f->get_unreferenced_variable_symbols().begin(),
12629 f->get_unreferenced_variable_symbols().end(),
12630 s->get_unreferenced_variable_symbols().begin(),
12631 s->get_unreferenced_variable_symbols().end(),
12632 r->priv_->unrefed_var_syms_edit_script_);
12634 if (ctxt->show_unreachable_types())
12637 diff_utils::compute_diff<type_base_wptr_it_type, eq_type>
12638 (f->get_types_not_reachable_from_public_interfaces().begin(),
12639 f->get_types_not_reachable_from_public_interfaces().end(),
12640 s->get_types_not_reachable_from_public_interfaces().begin(),
12641 s->get_types_not_reachable_from_public_interfaces().end(),
12642 r->priv_->unreachable_types_edit_script_);
12644 r->priv_->ensure_lookup_tables_populated();
12665 const corpus_group_sptr& s,
12669 corpus_sptr c1 = f;
12670 corpus_sptr c2 = s;
12681struct diff_node_visitor::priv
12683 diff* topmost_interface_diff;
12687 : topmost_interface_diff(),
12692 : topmost_interface_diff(),
12702diff_node_visitor::~diff_node_visitor() =
default;
12708 : priv_(new priv(k))
12718{
return priv_->kind;}
12738{priv_->kind = priv_->kind | v;}
12746{priv_->topmost_interface_diff = d;}
12754{
return priv_->topmost_interface_diff;}
12981 bool already_visited = d->
context()->diff_has_been_visited(d);
12989 bool update_canonical = !already_visited && canonical;
12991 for (vector<diff*>::const_iterator i = d->
children_nodes().begin();
13024 if (!already_visited && canonical)
13025 if (update_canonical)
13026 canonical->add_to_category(c);
13054 c &= (~NON_COMPATIBLE_NAME_CHANGE_CATEGORY
13055 & ~NON_COMPATIBLE_DISTINCT_CHANGE_CATEGORY);
13068 c &= (~NON_COMPATIBLE_NAME_CHANGE_CATEGORY
13069 & ~NON_COMPATIBLE_DISTINCT_CHANGE_CATEGORY);
13077 c &= ~NON_COMPATIBLE_NAME_CHANGE_CATEGORY;
13092 category_propagation_visitor v;
13093 bool s = diff_tree->
context()->visiting_a_node_twice_is_forbidden();
13094 diff_tree->
context()->forbid_visiting_a_node_twice(
true);
13095 diff_tree->
context()->forget_visited_diffs();
13097 diff_tree->
context()->forbid_visiting_a_node_twice(s);
13119 category_propagation_visitor v;
13120 bool s = diff_tree->
context()->visiting_a_node_twice_is_forbidden();
13121 diff_tree->
context()->forbid_visiting_a_node_twice(
false);
13123 diff_tree->
context()->forbid_visiting_a_node_twice(s);
13148 visit_begin(
diff* d)
13150 bool is_private_type =
false;
13161 if (canonical_diff != d)
13217 bool has_non_suppressed_child =
false;
13218 bool has_non_empty_child =
false;
13219 bool has_suppressed_child =
false;
13220 bool has_non_private_child =
false;
13221 bool has_private_child =
false;
13222 bool has_descendant_with_allowed_change =
false;
13255 && (!d->has_local_changes()
13280 for (vector<diff*>::const_iterator i = d->children_nodes().begin();
13281 i != d->children_nodes().end();
13285 if (child->has_changes())
13287 has_non_empty_child =
true;
13289 has_suppressed_child =
true;
13290 else if (child->get_class_of_equiv_category()
13296 has_non_suppressed_child =
true;
13298 if (child->get_class_of_equiv_category()
13300 has_private_child =
true;
13301 else if (child->get_class_of_equiv_category()
13307 has_non_private_child =
true;
13311 if (has_non_empty_child
13312 && has_suppressed_child
13313 && !has_non_suppressed_child)
13319 if (canonical_diff != d)
13335 if (has_non_empty_child
13336 && has_private_child
13337 && !has_non_private_child)
13343 if (canonical_diff != d)
13353 && has_private_child
13354 && has_non_empty_child)
13360 if (canonical_diff != d)
13377 if (fn_type_diff->is_suppressed())
13383 if (canonical_diff != d)
13392 for (
auto child_node : d->children_nodes())
13398 has_descendant_with_allowed_change =
true;
13400 if (has_descendant_with_allowed_change)
13403 d->add_to_category(c);
13404 d->get_canonical_diff()->add_to_category(c);
13418 if (diff_tree && !diff_tree->
context()->suppressions().empty())
13422 suppression_categorization_visitor v;
13423 diff_tree->
context()->forget_visited_diffs();
13424 bool s = diff_tree->
context()->visiting_a_node_twice_is_forbidden();
13425 diff_tree->
context()->forbid_visiting_a_node_twice(
true);
13427 diff_tree->
context()->forbid_visiting_a_node_twice(s);
13450 if (diff_tree && !diff_tree->
context()->suppressions().empty())
13455 suppression_categorization_visitor v;
13456 diff_tree->
context()->forget_visited_diffs();
13457 bool s = diff_tree->
context()->visiting_a_node_twice_is_forbidden();
13458 diff_tree->
context()->forbid_visiting_a_node_twice(
true);
13459 const_cast<corpus_diff*
>(diff_tree)->traverse(v);
13460 diff_tree->
context()->forbid_visiting_a_node_twice(s);
13466 apply_supprs_to_added_removed_fns_vars_unreachable_types();
13496 do_indent(
unsigned level)
13498 for (
unsigned i = 0; i < level; ++i)
13502 diff_node_printer(ostream& out)
13521 visit_begin(corpus_diff*)
13527 visit_end(corpus_diff*)
13533 visit(diff* d,
bool pre)
13542 out_ << d->get_pretty_representation();
13546 do_indent(level_ + 1);
13547 out_ <<
"category: "<< d->get_category() <<
"\n";
13548 do_indent(level_ + 1);
13549 out_ <<
"local category: "<< d->get_local_category() <<
"\n";
13550 do_indent(level_ + 1);
13551 out_ <<
"@: " << std::hex << d << std::dec <<
"\n";
13552 do_indent(level_ + 1);
13553 out_ <<
"@-canonical: " << std::hex
13554 << d->get_canonical_diff()
13555 << std::dec <<
"\n";
13563 visit(corpus_diff* d,
bool pre)
13572 for (
unsigned i = 0; i < level_; ++i)
13574 out_ << d->get_pretty_representation();
13593 diff_node_printer p(out);
13594 bool s = diff_tree->
context()->visiting_a_node_twice_is_forbidden();
13595 diff_tree->
context()->forbid_visiting_a_node_twice(
false);
13597 diff_tree->
context()->forbid_visiting_a_node_twice(s);
13611 diff_node_printer p(out);
13612 bool s = diff_tree->
context()->visiting_a_node_twice_is_forbidden();
13613 diff_tree->
context()->forbid_visiting_a_node_twice(
false);
13615 diff_tree->
context()->forbid_visiting_a_node_twice(s);
13650 std::cout << c << std::endl;
13663 bool skip_children_nodes_;
13665 redundancy_marking_visitor()
13666 : skip_children_nodes_()
13670 visit_begin(diff* d)
13672 if (d->to_be_reported())
13678 if ((d->context()->diff_has_been_visited(d)
13679 || d->get_canonical_diff()->is_traversing())
13680 && d->has_changes())
13695 bool redundant_with_sibling_node =
false;
13700 if (p &&
dynamic_cast<const fn_parm_diff*
>(p))
13704 for (vector<diff*>::const_iterator s =
13705 p->children_nodes().begin();
13706 s != p->children_nodes().end();
13714 if (fn_parm_diff* f =
dynamic_cast<fn_parm_diff*
>(*s))
13715 sib = f->type_diff().get();
13718 if (sib->get_canonical_diff() == d->get_canonical_diff()
13723 redundant_with_sibling_node =
true;
13727 if (!redundant_with_sibling_node
13748 && (!d->get_canonical_diff()->is_filtered_out()
13749 || (d->get_canonical_diff()->get_category()
13756 && d->context()->diff_has_been_visited(d) != d
13778 skip_children_nodes_ =
true;
13786 skip_children_nodes_ =
true;
13791 visit_begin(corpus_diff*)
13798 if (skip_children_nodes_)
13805 skip_children_nodes_ =
false;
13814 && ((!d->has_local_changes_to_be_reported()
13843 && (!(d->has_local_changes()
13854 && (!(d->has_local_changes()
13861 && (!(d->has_local_changes()
13865 bool has_non_redundant_child =
false;
13866 bool has_non_empty_child =
false;
13868 for (vector<diff*>::const_iterator i =
13869 d->children_nodes().begin();
13870 i != d->children_nodes().end();
13873 if ((*i)->has_changes())
13884 has_non_empty_child =
true;
13891 if ((*i)->to_be_reported()
13893 has_non_redundant_child =
true;
13895 if (has_non_redundant_child)
13902 if (has_non_empty_child
13903 && !has_non_redundant_child)
13910 visit_end(corpus_diff*)
13919 visit(corpus_diff*,
bool)
13927struct redundancy_clearing_visitor :
public diff_node_visitor
13930 visit(corpus_diff*,
bool)
13934 visit(diff* d,
bool)
13938 c &= ~REDUNDANT_CATEGORY;
13939 d->set_category(c);
13951 if (diff_tree->
context()->show_redundant_changes())
13953 redundancy_marking_visitor v;
13954 bool s = diff_tree->
context()->visiting_a_node_twice_is_forbidden();
13955 diff_tree->
context()->forbid_visiting_a_node_twice(
false);
13957 diff_tree->
context()->forbid_visiting_a_node_twice(s);
13975 redundancy_marking_visitor v;
13976 diff_tree->
context()->forget_visited_diffs();
13977 bool s = diff_tree->
context()->visiting_a_node_twice_is_forbidden();
13978 diff_tree->
context()->forbid_visiting_a_node_twice(
false);
13980 diff_tree->
context()->forbid_visiting_a_node_twice(s);
14000 redundancy_clearing_visitor v;
14001 bool s = diff_tree->
context()->visiting_a_node_twice_is_forbidden();
14002 diff_tree->
context()->forbid_visiting_a_node_twice(
false);
14004 diff_tree->
context()->forbid_visiting_a_node_twice(s);
14005 diff_tree->
context()->forget_visited_diffs();
14023 redundancy_clearing_visitor v;
14024 bool s = diff_tree->
context()->visiting_a_node_twice_is_forbidden();
14025 diff_tree->
context()->forbid_visiting_a_node_twice(
false);
14027 diff_tree->
context()->forbid_visiting_a_node_twice(s);
14028 diff_tree->
context()->forget_visited_diffs();
14054 if (!ctxt->perform_change_categorization())
14058 ctxt->maybe_apply_filters(diff_tree);
14072 c->apply_filters_and_suppressions_before_reporting();
14089 if (t && t->get_environment().is_variadic_parameter_type(t))
14093 if (t && t->get_environment().is_variadic_parameter_type(t))
14158 if (allow_indirect_type)
14359has_local_type_change_only(
const diff *d)
14387 return (has_local_type_change_only(v)
14390 return (has_local_type_change_only(p)
14394 return (has_local_type_change_only(f)
The private data and functions of the abigail::ir::comparison types.
#define SKIP_MEM_FN_IF_VIRTUALITY_DISALLOWED
Skip the processing of the current member function if its virtual-ness is disallowed by the user.
#define ABG_ASSERT(cond)
This is a wrapper around the 'assert' glibc call. It allows for its argument to have side effects,...
The abstraction of a diff between two arrays.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of array_diff.
const diff_sptr & element_type_diff() const
Getter for the diff between the two types of array elements.
array_diff(const array_type_def_sptr first, const array_type_def_sptr second, diff_sptr element_type_diff, vector< subrange_diff_sptr > &subrange_diffs, diff_context_sptr ctxt=diff_context_sptr())
Constructor for array_diff.
bool any_subrange_diff_to_be_reported() const
Test if any subrange diff is to be reported.
const array_type_def_sptr second_array() const
Getter for the second array of the diff.
const array_type_def_sptr first_array() const
Getter for the first array of the diff.
virtual enum change_kind has_local_changes() const
const vector< subrange_diff_sptr > & subrange_diffs() const
Getter for the diffs between the array subranges.
virtual const string & get_pretty_representation() const
virtual void report(ostream &, const string &indent="") const
Report the diff in a serialized form.
An abstraction of a diff between two instances of class_decl::base_spec.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of base_diff.
void set_underlying_class_diff(class_diff_sptr d)
Setter for the diff object for the diff of the underlyng base classes.
class_decl::base_spec_sptr second_base() const
Getter for the second base spec of the diff object.
base_diff(class_decl::base_spec_sptr first, class_decl::base_spec_sptr second, class_diff_sptr underlying, diff_context_sptr ctxt=diff_context_sptr())
const class_diff_sptr get_underlying_class_diff() const
Getter for the diff object for the diff of the underlying base classes.
class_decl::base_spec_sptr first_base() const
Getter for the first base spec of the diff object.
virtual enum change_kind has_local_changes() const
virtual const string & get_pretty_representation() const
virtual void report(ostream &, const string &indent="") const
Generates a report for the current instance of base_diff.
This type abstracts changes for a class_decl.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of class_diff.
class_decl_sptr first_class_decl() const
const base_diff_sptrs_type & changed_bases()
Getter for the changed base classes of the diff.
const vector< class_decl::base_spec_sptr > & moved_bases() const
Getter for the vector of bases that "moved". That is, the vector of base types which position changed...
const string_base_sptr_map & inserted_bases() const
Getter for the inserted base classes of the diff.
const string_base_sptr_map & deleted_bases() const
Getter for the deleted base classes of the diff.
virtual enum change_kind has_local_changes() const
const edit_script & base_changes() const
virtual const string & get_pretty_representation() const
class_diff(class_decl_sptr first_scope, class_decl_sptr second_scope, diff_context_sptr ctxt=diff_context_sptr())
Constructor of class_diff.
friend class_diff_sptr compute_diff(const class_decl_sptr first, const class_decl_sptr second, diff_context_sptr ctxt)
Compute the set of changes between two instances of class_decl.
class_decl_sptr second_class_decl() const
Getter of the second class involved in the diff.
virtual void report(ostream &, const string &indent="") const
Produce a basic report about the changes between two class_decl.
This is the base class of class_diff and union_diff.
virtual bool has_changes() const
Test if the current diff node carries a change.
const edit_script & member_fn_tmpls_changes() const
size_t count_filtered_subtype_changed_data_members(bool local_only=false) const
Count the number of /filtered/ data members with a sub-type change.
const class_or_union_diff::priv_ptr & get_priv() const
Getter of the private data of the class_or_union_diff type.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of class_or_un...
const edit_script & member_class_tmpls_changes() const
void allocate_priv_data()
Allocate the memory for the priv_ pimpl data member of the class_or_union_diff class.
class_or_union_diff(class_or_union_sptr first_scope, class_or_union_sptr second_scope, diff_context_sptr ctxt=diff_context_sptr())
Constructor for the class_or_union_diff class.
const unsigned_var_diff_sptr_map & changed_data_members() const
Getter of the map of data members that got replaced by another data member. The key of the map is the...
const edit_script & member_types_changes() const
const string_member_function_sptr_map & deleted_member_fns() const
class_or_union_sptr first_class_or_union() const
const var_diff_sptrs_type & sorted_subtype_changed_data_members() const
Getter of the sorted vector of data members with a (sub-)type change.
bool lookup_tables_empty(void) const
Tests if the lookup tables are empty.
const string_decl_base_sptr_map & data_members_replaced_by_adms() const
Get the map of data members that got replaced by anonymous data members.
void clear_lookup_tables(void)
Clear the lookup tables useful for reporting.
const string_decl_base_sptr_map & deleted_data_members() const
Getter for the data members that got deleted.
virtual ~class_or_union_diff()
Destructor of class_or_union_diff.
const string_decl_base_sptr_map & inserted_data_members() const
Getter for the data members that got inserted.
void ensure_lookup_tables_populated(void) const
If the lookup tables are not yet built, walk the differences and fill them.
const string_member_function_sptr_map & inserted_member_fns() const
const function_decl_diff_sptrs_type & changed_member_fns() const
Getter for the virtual members functions that have had a change in a sub-type, without having a chang...
const edit_script & data_members_changes() const
const changed_var_sptrs_type & ordered_data_members_replaced_by_adms() const
Get an ordered vector of of data members that got replaced by anonymous data members.
virtual enum change_kind has_local_changes() const
size_t count_filtered_changed_data_members(bool local_only=false) const
Count the number of /filtered/ data members that got replaced by another data member.
const edit_script & member_fns_changes() const
class_or_union_sptr second_class_or_union() const
const var_diff_sptrs_type & sorted_changed_data_members() const
Getter of the sorted vector of data members that got replaced by another data member.
virtual void report(ostream &, const string &indent="") const
Report the changes carried by the current class_or_union_diff node in a textual format.
This is a document class that aims to capture statistics about the changes carried by a corpus_diff t...
size_t num_changed_unreachable_types_filtered_out() const
Getter of the number of changed types that are unreachable from public interfaces and that have been ...
size_t num_func_removed() const
Getter for the number of functions removed.
size_t num_removed_unreachable_types_filtered_out() const
Getter of the number of removed types that are not reachable from public interfaces and that have bee...
size_t num_vars_changed() const
Getter for the number of variables that have a change in one of their sub-types.
size_t num_var_with_incompatible_changes() const
Getter for the number of variables with incompatible changes.
size_t net_num_leaf_func_non_incompatible_changes() const
Getter for the net number of leaf function diff nodes that carry changes that are NOT incompatible.
size_t net_num_leaf_var_changes() const
Getter for the net number of leaf variable change diff nodes.
size_t num_vars_added() const
Getter for the number of variables added.
size_t num_removed_unreachable_types() const
Getter of the number of removed types that are unreachable from the public interface of the ABI corpu...
size_t num_changed_vars_filtered_out() const
Getter for the number of variables that have a change in one of their sub-types, and that have been f...
size_t num_changed_func_filtered_out() const
Getter for the number of functions that have a change in one of their sub-types, and that have been f...
size_t net_num_non_incompatible_var_changed() const
Getter of the net number of variables with changes that are not incompatible.
size_t num_removed_vars_filtered_out() const
Getter for the number removed variables that have been filtered out.
size_t net_num_func_changed() const
Getter for the number of functions that have a change in their sub-types, minus the number of these f...
size_t net_num_vars_removed() const
Getter for the net number of removed variables.
size_t net_num_added_unreachable_types() const
Getter of the number of added types that are unreachable from public interfaces and that are *NOT* fi...
size_t num_removed_var_syms_filtered_out() const
Getter for the number of removed variable symbols, not referenced by any debug info,...
size_t num_leaf_func_with_incompatible_changes() const
Getter for the number of leaf function diff nodes that carry incompatible changes.
size_t net_num_added_func_syms() const
Getter of the net number of added function symbols that are not referenced by any debug info.
size_t num_added_var_syms_filtered_out() const
Getter for the number of added variable symbols, not referenced by any debug info,...
size_t num_added_unreachable_types() const
Getter of the number of added types that are unreachable from the public interface of the ABI corpus.
size_t num_func_with_local_harmful_changes() const
Getter for the number of functions with local harmful changes.
size_t net_num_removed_func_syms() const
Getter of the net number of removed function symbols that are not referenced by any debug info.
size_t num_var_syms_added() const
Getter for the number of variable symbols (not referenced by any debug info) that got added.
size_t num_leaf_var_changes() const
Getter for the number of leaf variable change diff nodes.
size_t net_num_removed_var_syms() const
Getter of the net number of removed variable symbols that are not referenced by any debug info.
size_t num_func_with_incompatible_changes() const
Getter for the number of functions with incompatible changes.
size_t net_num_non_incompatible_func_changed() const
Getter of the net number of functions with changes that are not incompatible.
size_t net_num_func_removed() const
Getter for the net number of function removed.
size_t net_num_func_added() const
Getter for the net number of added functions.
size_t net_num_removed_unreachable_types() const
Getter of the number of removed types that are not reachable from public interfaces and that have *NO...
size_t net_num_leaf_changes() const
Getter of the net number of leaf change diff nodes.
size_t num_removed_func_syms_filtered_out() const
Getter for the number of removed function symbols, not referenced by debug info, that have been filte...
size_t num_added_unreachable_types_filtered_out() const
Getter of the number of added types that are unreachable from public interfaces and that are filtered...
size_t num_added_func_filtered_out() const
Getter for the number of added function that have been filtered out.
size_t num_func_syms_added() const
Getter for the number of function symbols (not referenced by any debug info) that got added.
size_t net_num_leaf_type_changes() const
Getter for the net number of leaf type change diff nodes.
size_t num_func_added() const
Getter for the number of functions added.
size_t net_num_added_var_syms() const
Getter of the net number of added variable symbols that are not referenced by any debug info.
size_t num_leaf_type_changes() const
Getter for the number of leaf type change diff nodes.
size_t num_leaf_var_changes_filtered_out() const
Getter for the number of leaf variable changes diff nodes that have been filtered out.
size_t num_added_vars_filtered_out() const
Getter for the number of added variables that have been filtered out.
size_t num_func_with_virtual_offset_changes() const
Getter for the number of functions that carry virtual member offset changes.
size_t net_num_leaf_var_non_incompatible_changes() const
Getter for the net number of leaf variable diff nodes that carry changes that are NOT incompatible.
size_t num_func_changed() const
Getter for the number of functions that have a change in one of their sub-types.
size_t num_removed_func_filtered_out() const
Getter for the number of removed functions that have been filtered out.
size_t net_num_vars_added() const
Getter for the net number of added variables.
size_t num_leaf_changes() const
Getter of the number of leaf type change diff nodes.
size_t num_leaf_func_changes_filtered_out() const
Getter for the number of leaf function change diff nodes that were filtered out.
size_t num_added_func_syms_filtered_out() const
Getter for the number of added function symbols, not referenced by any debug info,...
size_t num_leaf_type_changes_filtered_out() const
Getter for the number of filtered out leaf type change diff nodes.
size_t num_changed_unreachable_types() const
Getter of the number of changed types that are unreachable from the public interface of the ABI corpu...
size_t net_num_leaf_func_changes() const
Getter for the net number of leaf function change diff nodes.
size_t num_leaf_func_changes() const
Getter for the number of leaf function change diff nodes.
size_t net_num_changed_unreachable_types() const
Getter of the number of changed types that are unreachable from public interfaces and that have *NOT*...
size_t num_func_syms_removed() const
Getter for the number of function symbols (not referenced by any debug info) that got removed.
size_t num_leaf_var_with_incompatible_changes() const
Getter for the number of leaf variable diff nodes that carry incompatible changes.
size_t num_leaf_changes_filtered_out() const
Getter of the number of leaf type change diff nodes that have been filtered out.
size_t num_vars_removed() const
Getter for the number of variables removed.
size_t num_var_syms_removed() const
Getter for the number of variable symbols (not referenced by any debug info) that got removed.
size_t num_var_with_local_harmful_changes() const
Getter for the number of variables with local harmful changes.
size_t net_num_vars_changed() const
Getter for the number of variables that have a change in their sub-types, minus the number of these v...
An abstraction of a diff between between two abi corpus.
bool has_incompatible_changes() const
Test if the current instance of corpus_diff carries changes that we are sure are incompatible....
bool has_changes() const
Return true iff the current corpus_diff node carries a change.
void finish_diff_type()
Finish building the current instance of corpus_diff.
virtual void chain_into_hierarchy()
Populate the vector of children node of the corpus_diff type.
const string_var_ptr_map & deleted_variables() const
Getter for the variables that got deleted from the first subject of the diff.
const vector< diff_sptr > & changed_unreachable_types_sorted() const
Getter of a sorted vector of changed types that are not reachable from global functions/variables.
bool soname_changed() const
Test if the soname of the underlying corpus has changed.
friend corpus_diff_sptr compute_diff(const corpus_sptr f, const corpus_sptr s, diff_context_sptr ctxt)
Compute the diff between two instances of corpus.
const vector< type_base_sptr > & deleted_unreachable_types_sorted() const
Getter of a sorted vector of deleted types that are not reachable from global functions/variables.
edit_script & function_changes() const
edit_script & variable_changes() const
const vector< diff * > & children_nodes() const
const string_diff_sptr_map & changed_unreachable_types() const
Getter for a map of changed types that are not reachable from global functions/variables.
const var_diff_sptrs_type & changed_variables_sorted()
Getter for the sorted vector of variables which signature didn't change but which do have some indire...
const string_elf_symbol_map & deleted_unrefed_function_symbols() const
Getter for function symbols not referenced by any debug info and that got deleted.
const string_elf_symbol_map & deleted_unrefed_variable_symbols() const
Getter for variable symbols not referenced by any debug info and that got deleted.
const string_function_decl_diff_sptr_map & changed_functions() const
Getter for the functions which signature didn't change, but which do have some indirect changes in th...
const var_diff_sptrs_type & incompatible_changed_variables() const
Getter of the set of diff nodes representing incompatibly changed global variables.
corpus_diff(corpus_sptr first, corpus_sptr second, diff_context_sptr ctxt=diff_context_sptr())
Constructor for corpus_diff.
corpus_sptr second_corpus() const
bool do_log() const
Test if logging was requested.
const string_elf_symbol_map & added_unrefed_function_symbols() const
Getter for function symbols not referenced by any debug info and that got added.
bool has_net_subtype_changes() const
Test if the current instance of corpus_diff carries subtype changes whose reports are not suppressed ...
const string_var_ptr_map & added_variables() const
Getter for the added variables of the diff.
diff_maps & get_leaf_diffs()
Get the set of maps that contain leaf nodes. A leaf node being a node with a local change.
const diff_context_sptr context() const
Getter of the diff context of this diff.
bool has_net_changes() const
Test if the current instance of corpus_diff carries changes whose reports are not suppressed by any s...
virtual bool traverse(diff_node_visitor &v)
Traverse the diff sub-tree under the current instance corpus_diff.
const diff_stats & apply_filters_and_suppressions_before_reporting()
Apply the different filters that are registered to be applied to the diff tree; that includes the cat...
const string_var_diff_sptr_map & changed_variables()
Getter for the non-sorted map of variables which signature didn't change but which do have some indir...
const string_function_ptr_map & added_functions()
Getter for the added functions of the diff.
void mark_leaf_diff_nodes()
Walks the diff nodes associated to the current corpus diff and mark those that carry local changes....
const string_type_base_sptr_map & deleted_unreachable_types() const
Getter for a map of deleted types that are not reachable from global functions/variables.
corpus_sptr first_corpus() const
const vector< type_base_sptr > & added_unreachable_types_sorted() const
Getter of a sorted vector of added types that are not reachable from global functions/variables.
const function_decl_diff_sptrs_type & incompatible_changed_functions() const
Getter of the set of diff nodes representing incompatibly changed functions.
virtual void report(ostream &out, const string &indent="") const
Report the diff in a serialized form.
const string_elf_symbol_map & added_unrefed_variable_symbols() const
Getter for variable symbols not referenced by any debug info and that got added.
const string & get_pretty_representation() const
const string_function_ptr_map & deleted_functions() const
Getter for the deleted functions of the diff.
void append_child_node(diff_sptr)
Append a new child node to the vector of children nodes for the current instance of corpus_diff node.
friend void apply_suppressions(const corpus_diff *diff_tree)
Walk a corpus_diff tree and appply the suppressions carried by the context. If the suppression applie...
const function_decl_diff_sptrs_type & changed_functions_sorted() const
Getter for a sorted vector of functions which signature didn't change, but which do have some indirec...
const string_type_base_sptr_map & added_unreachable_types() const
Getter for a map of added types that are not reachable from global functions/variables.
bool architecture_changed() const
Test if the architecture of the underlying corpus has changed.
The base class of diff between decls.
decl_diff_base(decl_base_sptr first_subject, decl_base_sptr second_subject, diff_context_sptr ctxt)
Constructor of decl_diff_base.
The default, initial, reporter of the libabigail comparison engine.
The context of the diff. This type holds various bits of information that is going to be used through...
bool show_deleted_vars() const
void add_suppressions(const suppr::suppressions_type &supprs)
Add new suppression specifications that specify which diff node reports should be dropped on the floo...
diff_category get_allowed_category() const
Getter for the bitmap that represents the set of categories that the user wants to see reported.
void forget_visited_diffs()
Unmark all the diff nodes that were marked as being traversed.
corpus_sptr get_first_corpus() const
Getter for the first corpus of the corpus diff of the current context.
bool show_architecture_change() const
Getter for the property that says if the comparison module should show the architecture changes in it...
bool show_offsets_sizes_in_bits() const
Get the flag that indicates if diff reports using this context should show sizes and offsets in bits,...
void forbid_visiting_a_node_twice(bool f)
This sets a flag that, if it's true, then during the traversing of a diff nodes tree each node is vis...
bool show_changed_fns() const
void initialize_canonical_diff(const diff_sptr diff)
Set the canonical diff node property of a given diff node appropriately.
bool show_redundant_changes() const
A getter for the flag that says if we should report about functions or variables diff nodes that have...
void forbid_visiting_a_node_twice_per_interface(bool)
This function sets a flag os that if forbid_visiting_a_node_twice() returns true, then each time the ...
void keep_diff_alive(diff_sptr &)
Add a diff node to the set of diff nodes that are kept alive for the life time of the current instanc...
diff * diff_has_been_visited(const diff *) const
Test if a diff node has been traversed.
bool visiting_a_node_twice_is_forbidden_per_interface() const
Return a flag that, if true, then during the traversing of a diff nodes tree each node is visited at ...
void set_corpus_diff(const corpus_diff_sptr &)
Set the corpus diff relevant to this context.
bool show_leaf_changes_only() const
Get the flag that indicates if the diff using this context should show only leaf changes or not.
bool perform_change_categorization() const
Test if it's requested to perform diff node categorization.
bool show_impacted_interfaces() const
Getter of the flag that indicates if the leaf reporter should display a summary of the interfaces imp...
bool show_soname_change() const
Getter for the property that says if the comparison module should show the soname changes in its repo...
reporter_base_sptr get_reporter() const
Getter of the reporter to be used in this context.
void add_diff_filter(filtering::filter_base_sptr)
Setter for the diff filters to apply to a given diff sub-tree.
bool do_log() const
Test if logging was requested.
const suppr::suppressions_type & direct_suppressions() const
Getter of the direct suppression specification (those that are not negated) comprised in the general ...
void maybe_apply_filters(diff_sptr diff)
Apply the diff filters to a given diff sub-tree.
bool show_added_fns() const
const suppr::suppressions_type & suppressions() const
Getter for the vector of suppressions that specify which diff node reports should be dropped on the f...
bool show_relative_offset_changes(void)
Get the flag saying if offset changes should be reported in a relative way. That is,...
bool visiting_a_node_twice_is_forbidden() const
Return a flag that, if true, then during the traversing of a diff nodes tree each node is visited at ...
void do_dump_diff_tree(const diff_sptr) const
Emit a textual representation of a diff tree to the error output stream of the current context,...
const suppr::suppressions_type & negated_suppressions() const
Getter of the negated suppression specifications that are comprised in the general vector of suppress...
void add_suppression(const suppr::suppression_sptr suppr)
Add a new suppression specification that specifies which diff node reports should be dropped on the f...
bool show_hex_values() const
Get the flag that indicates if the diff reports using this context should show sizes and offsets in a...
bool show_deleted_fns() const
void switch_categories_off(diff_category c)
Setter for the bitmap that represents the set of categories that the user wants to see reported.
bool show_stats_only() const
Test if the comparison module should only show the diff stats.
bool show_added_vars() const
const filtering::filters & diff_filters() const
Getter for the diff tree nodes filters to apply to diff sub-trees.
bool show_unreachable_types()
Getter for the flag that indicates if changes on types unreachable from global functions and variable...
const corpus_diff_sptr & get_corpus_diff() const
Get the corpus diff for the current context.
void mark_diff_as_visited(const diff *)
Mark a diff node as traversed by a traversing algorithm.
diff_sptr get_canonical_diff_for(const type_or_decl_base_sptr first, const type_or_decl_base_sptr second) const
Getter for the canonical diff node for the diff represented by their two subjects.
bool show_changed_vars() const
void switch_categories_on(diff_category c)
Setter for the bitmap that represents the set of categories that the user wants to see reported.
ostream * default_output_stream()
Getter for the default output stream used by code of the comparison engine. By default the default ou...
bool dump_diff_tree() const
Test if the comparison engine should dump the diff tree for the changed functions and variables it ha...
bool show_symbols_unreferenced_by_debug_info() const
Getter for the flag that indicates if symbols not referenced by any debug info are to be compared and...
bool show_added_symbols_unreferenced_by_debug_info() const
Getter for the flag that indicates if symbols not referenced by any debug info and that got added are...
corpus_sptr get_second_corpus() const
Getter for the second corpus of the corpus diff of the current context.
void set_allowed_category(diff_category c)
Setter for the bitmap that represents the set of categories that the user wants to see reported.
void set_reporter(reporter_base_sptr &)
Setter of the reporter to be used in this context.
ostream * error_output_stream() const
Getter for the errror output stream used by code of the comparison engine. By default the error outpu...
This type contains maps. Each map associates a type name to a diff of that type. Not all kinds of dif...
const string_diff_ptr_map & get_function_decl_diff_map() const
Getter of the map that contains function decl diffs.
const string_diff_ptr_map & get_var_decl_diff_map() const
Getter of the map that contains var decl diffs.
const string_diff_ptr_map & get_enum_diff_map() const
Getter of the map that contains enum type diffs.
bool insert_diff_node(const diff *d, const type_or_decl_base_sptr &impacted_iface)
Insert a new diff node into the current instance of diff_maps.
diff_maps()
Default constructor of the diff_maps type.
const string_diff_ptr_map & get_union_diff_map() const
Getter of the map that contains union type diffs.
artifact_sptr_set_type * lookup_impacted_interfaces(const diff *d) const
Lookup the interfaces that are impacted by a given leaf diff node.
const string_diff_ptr_map & get_function_type_diff_map() const
Getter of the map that contains function type diffs.
const string_diff_ptr_map & get_typedef_diff_map() const
Getter of the map that contains typedef type diffs.
const string_diff_ptr_map & get_distinct_diff_map() const
Getter of the map that contains distinct diffs.
const string_diff_ptr_map & get_subrange_diff_map() const
Getter of the map that contains subrange type diffs.
const string_diff_ptr_map & get_reference_diff_map() const
Getter of the map that contains reference type diffs.
const string_diff_ptr_map & get_array_diff_map() const
Getter of the map that contains array type diffs.
const string_diff_ptr_map & get_type_decl_diff_map() const
Getter of the map that contains basic type diffs.
const string_diff_ptr_map & get_fn_parm_diff_map() const
Getter of the map that contains function parameter diffs.
const string_diff_ptr_map & get_class_diff_map() const
Getter of the map that contains class type diffs.
The base class for the node visitors. These are the types used to visit each node traversed by the di...
void or_visiting_kind(visiting_kind v)
Setter for the visiting policy of the traversing code while invoking this visitor....
virtual bool visit(diff *, bool)
Default visitor implementation.
virtual bool visit(distinct_diff *, bool)
Default visitor implementation.
virtual void visit_end(corpus_diff *)
This is called by the traversing code on a corpus_diff node just after visiting it....
void set_current_topmost_iface_diff(diff *)
Setter of the diff current topmost interface which is impacted by the current diff node being visited...
virtual void visit_begin(diff *)
This is called by the traversing code on a diff node just before visiting it. That is,...
visiting_kind get_visiting_kind() const
Getter for the visiting policy of the traversing code while invoking this visitor.
virtual void visit_end(diff *)
This is called by the traversing code on a diff node just after visiting it. That is after visiting i...
diff_node_visitor()
Default constructor of the diff_node_visitor type.
void set_visiting_kind(visiting_kind v)
Setter for the visiting policy of the traversing code while invoking this visitor.
diff * get_current_topmost_iface_diff() const
Getter of the diff current topmost interface which is impacted by the current diff node being visited...
virtual bool traverse(diff_node_visitor &v)
The default traverse function.
The abstraction of a change between two ABI artifacts, a.k.a an artifact change.
void begin_traversing()
Flag a given diff node as being traversed.
void set_category(diff_category c)
Set the category of the current diff node. This category includes the categories inherited from the c...
virtual void finish_diff_type()
Finish the insertion of a diff tree node into the diff graph.
virtual void chain_into_hierarchy()
This constructs the relation between this diff node and its detail diff nodes, in the generic view of...
diff_category remove_from_category(diff_category c)
Remove the current diff tree node from an a existing sef of categories. The categories include those ...
type_or_decl_base_sptr second_subject() const
Getter of the second subject of the diff.
bool is_traversing() const
Tell if a given node is being traversed or not.
type_or_decl_base_sptr first_subject() const
Getter of the first subject of the diff.
bool is_suppressed() const
Test if the current diff node has been suppressed by a user-provided suppression specification.
bool is_filtered_out_without_looking_at_allowed_changes() const
Test if this diff tree node is to be filtered out for reporting purposes, but without considering the...
diff * get_canonical_diff() const
Getter for the canonical diff of the current instance of diff.
bool has_parent_allowed_by_specific_negated_suppression() const
Test if the current diff node has a parent node which is specifically allowed by a negated suppressio...
bool has_local_changes_to_be_reported() const
Test if this diff tree node should be reported when considering the categories that were *NOT* inheri...
const vector< diff * > & children_nodes() const
Getter for the children nodes of the current diff node.
diff_category get_category() const
Getter for the category of the current diff tree node.
bool is_allowed_by_specific_negated_suppression() const
Test if this diff node is allowed (prevented from being suppressed) by at least one negated suppressi...
diff_category remove_from_local_category(diff_category c)
Remove the current diff tree node from the categories resulting from the local changes.
void add_to_local_and_inherited_categories(diff_category c)
Adds the current diff tree node to the categories resulting from the local and inherited changes of t...
bool do_log() const
Test if logging was requested.
diff_category get_local_category() const
Getter for the local category of the current diff tree node.
diff_category add_to_category(diff_category c)
Adds the current diff tree node to an additional set of categories. Note that the categories include ...
bool is_categorized_as_suppressed() const
Test if the current diff node has been suppressed by a suppression specification or it has been categ...
const diff_context_sptr context() const
Getter of the context of the current diff.
virtual enum change_kind has_local_changes() const =0
Pure interface to know if the current instance of @diff carries a local change. A local change is a c...
virtual bool traverse(diff_node_visitor &v)
The generic traversing code that walks a given diff sub-tree.
bool currently_reporting() const
Tests if we are currently in the middle of emitting a report for this diff.
virtual bool has_changes() const =0
Pure interface to get the length of the changes encapsulated by this diff. A length of zero means tha...
bool has_descendant_allowed_by_specific_negated_suppression() const
Test if the current diff node has a descendant node which is specifically allowed by a negated suppre...
bool to_be_reported() const
Test if this diff tree node should be reported.
const diff * parent_node() const
Getter for the parent node of the current diff node.
diff_category get_class_of_equiv_category() const
Getter of the category of the class of equivalence of the current diff tree node.
void set_canonical_diff(diff *)
Setter for the canonical diff of the current instance of diff.
bool is_filtered_out_wrt_non_inherited_categories() const
Test if this diff tree node is to be filtered out for reporting purposes, but by considering only the...
diff_category add_to_local_category(diff_category c)
Adds the current diff tree node to the categories resulting from the local changes of the current dif...
void set_local_category(diff_category c)
Set the local category of the current diff node.
virtual const string & get_pretty_representation() const
Get a pretty representation of the current diff node.
void append_child_node(diff_sptr)
Add a new child node to the vector of children nodes for the current diff node.
bool is_filtered_out() const
Test if this diff tree node is to be filtered out for reporting purposes.
void end_traversing()
Flag a given diff node as not being traversed anymore.
bool reported_once() const
Tests if a report has already been emitted for the current diff.
An abstraction of a diff between entities that are of a different kind (disctinct).
virtual bool has_changes() const
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of @distinct_d...
const diff_sptr compatible_child_diff() const
Getter for the child diff of this distinct_diff instance.
distinct_diff(type_or_decl_base_sptr first, type_or_decl_base_sptr second, diff_context_sptr ctxt=diff_context_sptr())
Constructor for distinct_diff.
static bool entities_are_of_distinct_kinds(type_or_decl_base_sptr first, type_or_decl_base_sptr second)
Test if the two arguments are of different kind, or that are both NULL.
virtual void report(ostream &out, const string &indent="") const
Emit a report about the current diff instance.
const type_or_decl_base_sptr first() const
Getter for the first subject of the diff.
virtual enum change_kind has_local_changes() const
virtual const string & get_pretty_representation() const
const type_or_decl_base_sptr second() const
Getter for the second subject of the diff.
Abstraction of a diff between two enums.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of enum_diff.
enum_diff(const enum_type_decl_sptr, const enum_type_decl_sptr, const diff_sptr, diff_context_sptr ctxt=diff_context_sptr())
Constructor for enum_diff.
diff_sptr underlying_type_diff() const
const string_changed_enumerator_map & changed_enumerators() const
virtual enum change_kind has_local_changes() const
virtual const string & get_pretty_representation() const
const enum_type_decl_sptr first_enum() const
const string_enumerator_map & deleted_enumerators() const
const enum_type_decl_sptr second_enum() const
const string_enumerator_map & inserted_enumerators() const
virtual void report(ostream &, const string &indent="") const
Report the differences between the two enums.
A filter that walks the diff nodes tree and tags relevant diff nodes into categories considered to re...
Abstraction of a diff between two function parameters.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
virtual void chain_into_hierarchy()
Populate the vector of children nodes of the diff base type sub-object of this instance of fn_parm_di...
const function_decl::parameter_sptr first_parameter() const
Getter for the first subject of this diff node.
virtual enum change_kind has_local_changes() const
Check if the current diff node carries a local change.
virtual const string & get_pretty_representation() const
Build and return a textual representation of the current instance of fn_parm_diff.
const function_decl::parameter_sptr second_parameter() const
Getter for the second subject of this diff node.
diff_sptr type_diff() const
Getter for the diff representing the changes on the type of the function parameter involved in the cu...
virtual void report(ostream &, const string &indent="") const
Emit a textual report about the current fn_parm_diff instance.
Abstraction of a diff between two function_decl.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of function_de...
const function_decl_sptr second_function_decl() const
function_decl_diff(const function_decl_sptr first, const function_decl_sptr second, diff_context_sptr ctxt)
Constructor for function_decl_diff.
virtual enum change_kind has_local_changes() const
const function_decl_sptr first_function_decl() const
virtual const string & get_pretty_representation() const
virtual void report(ostream &, const string &indent="") const
Serialize a report of the changes encapsulated in the current instance of function_decl_diff over to ...
Abstraction of a diff between two function types.
virtual bool has_changes() const
Test if the current diff node carries changes.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of function_ty...
const string_fn_parm_diff_sptr_map & subtype_changed_parms() const
Getter for the map of function parameter changes of the current diff.
const diff_sptr return_type_diff() const
Getter for the diff of the return types of the two function types of the current diff.
const string_parm_map & removed_parms() const
Getter for the map of parameters that got removed.
const string_parm_map & added_parms() const
Getter for the map of parameters that got added.
friend function_type_diff_sptr compute_diff(const function_type_sptr first, const function_type_sptr second, diff_context_sptr ctxt)
Compute the diff between two instances of function_type.
const vector< function_decl::parameter_sptr > & sorted_added_parms() const
Getter for the sorted vector of added parameters .
const function_type_sptr first_function_type() const
Getter for the first subject of the diff.
const function_type_sptr second_function_type() const
Getter for the second subject of the diff.
function_type_diff(const function_type_sptr first, const function_type_sptr second, diff_context_sptr ctxt)
Consutrctor of the function_type type.
virtual enum change_kind has_local_changes() const
Test if the current diff node carries local changes.
virtual const string & get_pretty_representation() const
Build and return a copy of a pretty representation of the current instance of function_type_diff.
const vector< function_decl::parameter_sptr > & sorted_deleted_parms() const
Getter for the sorted vector of deleted parameters.
virtual void report(ostream &, const string &indent="") const
Build and emit a textual report about the current function_type_diff instance.
A reporter that only reports leaf changes.
The abstraction of a diff between two pointers.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
const pointer_type_def_sptr first_pointer() const
Getter for the first subject of a pointer diff.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of pointer_dif...
const pointer_type_def_sptr second_pointer() const
Getter for the second subject of a pointer diff.
diff_sptr underlying_type_diff() const
Getter for the diff between the pointed-to types of the pointers of this diff.
pointer_diff(pointer_type_def_sptr first, pointer_type_def_sptr second, diff_sptr underlying_type_diff, diff_context_sptr ctxt=diff_context_sptr())
Constructor for a pointer_diff.
virtual enum change_kind has_local_changes() const
virtual const string & get_pretty_representation() const
virtual void report(ostream &, const string &indent="") const
Report the diff in a serialized form.
The abstraction of a diff between two ptr_to_mbr_type.
virtual bool has_changes() const
Test whether the current diff node carries any change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of ptr_to_mbr_...
virtual ~ptr_to_mbr_diff()
Destructor of ptr_to_mbr_diff.
ptr_to_mbr_type_sptr first_ptr_to_mbr_type() const
Getter of the first pointer-to-member subject of the current diff node.
const diff_sptr containing_type_diff() const
Getter of the diff node carrying changes to the containing type of first subject of the current diff ...
const diff_sptr member_type_diff() const
Getter of the diff node carrying changes to the member type of first subject of the current diff node...
ptr_to_mbr_type_sptr second_ptr_to_mbr_type() const
Getter of the second pointer-to-member subject of the current diff node.
virtual enum change_kind has_local_changes() const
Test whether the current diff node carries any local change.
virtual const string & get_pretty_representation() const
Get the pretty representation of the current ptr_to_mbr_diff node.
virtual void report(ostream &, const string &indent="") const
Pure interface to report the diff in a serialized form that is legible for the user.
Abstraction of a diff between two qualified types.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of qualified_t...
diff_sptr leaf_underlying_type_diff() const
Getter for the diff between the most underlying non-qualified types of two qualified types.
diff_sptr underlying_type_diff() const
Getter for the diff between the underlying types of the two qualified types.
qualified_type_diff(qualified_type_def_sptr first, qualified_type_def_sptr second, diff_sptr underling, diff_context_sptr ctxt=diff_context_sptr())
Constructor for qualified_type_diff.
const qualified_type_def_sptr second_qualified_type() const
Getter for the second qualified type of the diff.
virtual enum change_kind has_local_changes() const
const qualified_type_def_sptr first_qualified_type() const
Getter for the first qualified type of the diff.
virtual const string & get_pretty_representation() const
virtual void report(ostream &, const string &indent="") const
Report the diff in a serialized form.
The abstraction of a diff between two references.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of reference_d...
reference_type_def_sptr first_reference() const
Getter for the first reference of the diff.
reference_type_def_sptr second_reference() const
Getter for the second reference of the diff.
reference_diff(const reference_type_def_sptr first, const reference_type_def_sptr second, diff_sptr underlying, diff_context_sptr ctxt=diff_context_sptr())
Constructor for reference_diff.
virtual enum change_kind has_local_changes() const
virtual const string & get_pretty_representation() const
const diff_sptr & underlying_type_diff() const
Getter for the diff between the two referred-to types.
virtual void report(ostream &, const string &indent="") const
Report the diff in a serialized form.
An abstractions of the changes between two scopes.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of scope_diff.
const diff_sptrs_type & changed_types() const
const scope_decl_sptr second_scope() const
Getter for the second scope of the diff.
const scope_decl_sptr first_scope() const
Getter for the first scope of the diff.
const decl_base_sptr deleted_member_at(unsigned index) const
Accessor that eases the manipulation of the edit script associated to this instance....
const diff_sptrs_type & changed_decls() const
scope_diff(scope_decl_sptr first_scope, scope_decl_sptr second_scope, diff_context_sptr ctxt=diff_context_sptr())
Constructor for scope_diff.
friend scope_diff_sptr compute_diff(const scope_decl_sptr first, const scope_decl_sptr second, scope_diff_sptr d, diff_context_sptr ctxt)
Compute the diff between two scopes.
const decl_base_sptr inserted_member_at(unsigned i)
Accessor that eases the manipulation of the edit script associated to this instance....
virtual void report(ostream &out, const string &indent="") const
Report the changes of one scope against another.
virtual enum change_kind has_local_changes() const
virtual const string & get_pretty_representation() const
const edit_script & member_changes() const
Accessor of the edit script of the members of a scope.
The abstraction of the diff between two subrange types.
virtual bool has_changes() const
Test if the current subrange_diff node carries any change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of subrange_di...
const array_type_def::subrange_sptr second_subrange() const
Getter of the second subrange of the current instance subrange_diff.
subrange_diff(const array_type_def::subrange_sptr &first, const array_type_def::subrange_sptr &second, const diff_sptr &underlying_type_diff, const diff_context_sptr ctxt=diff_context_sptr())
Constructor of the subrange_diff diff node type.
const array_type_def::subrange_sptr first_subrange() const
Getter of the first subrange of the current instance subrange_diff.
const diff_sptr underlying_type_diff() const
Getter of the diff node of the underlying types of the current subrange_diff diff node.
virtual enum change_kind has_local_changes() const
Test if the current subrange_diff node carries any local change.
virtual const string & get_pretty_representation() const
Getter the pretty representation of the subrange_diff diff node.
virtual void report(ostream &, const string &indent="") const
Report about the changes carried by this node.
An abstraction of a diff between two translation units.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
const translation_unit_sptr second_translation_unit() const
Getter for the second translation unit of this diff.
translation_unit_diff(translation_unit_sptr first, translation_unit_sptr second, diff_context_sptr ctxt=diff_context_sptr())
Constructor for translation_unit_diff.
virtual void report(ostream &out, const string &indent="") const
Report the diff in a serialized form.
virtual enum change_kind has_local_changes() const
const translation_unit_sptr first_translation_unit() const
Getter for the first translation unit of this diff.
Abstraction of a diff between two basic type declarations.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
const type_decl_sptr first_type_decl() const
Getter for the first subject of the type_decl_diff.
const type_decl_sptr second_type_decl() const
Getter for the second subject of the type_decl_diff.
virtual void report(ostream &out, const string &indent="") const
Ouputs a report of the differences between of the two type_decl involved in the type_decl_diff.
virtual enum change_kind has_local_changes() const
virtual const string & get_pretty_representation() const
The base class of diff between types.
Abstraction of a diff between two typedef_decl.
virtual bool has_changes() const
Return true iff the current diff node carries a change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of typedef_dif...
const typedef_decl_sptr second_typedef_decl() const
Getter for the second typedef_decl involved in the diff.
const typedef_decl_sptr first_typedef_decl() const
Getter for the firt typedef_decl involved in the diff.
const diff_sptr underlying_type_diff() const
Getter for the diff between the two underlying types of the typedefs.
virtual enum change_kind has_local_changes() const
virtual const string & get_pretty_representation() const
virtual void report(ostream &, const string &indent="") const
Reports the difference between the two subjects of the diff in a serialized form.
union_diff(union_decl_sptr first_union, union_decl_sptr second_union, diff_context_sptr ctxt=diff_context_sptr())
Constructor for the union_diff type.
union_decl_sptr first_union_decl() const
union_decl_sptr second_union_decl() const
virtual ~union_diff()
Destructor of the union_diff node.
virtual const string & get_pretty_representation() const
virtual void report(ostream &, const string &indent="") const
Report the changes carried by the current union_diff node in a textual format.
Abstracts a diff between two instances of var_decl.
virtual bool has_changes() const
Return true iff the diff node has a change.
virtual void chain_into_hierarchy()
Populate the vector of children node of the diff base type sub-object of this instance of var_diff.
var_diff(var_decl_sptr first, var_decl_sptr second, diff_sptr type_diff, diff_context_sptr ctxt=diff_context_sptr())
Constructor for var_diff.
var_decl_sptr first_var() const
Getter for the first var_decl of the diff.
virtual void report(ostream &out, const string &indent="") const
Report the diff in a serialized form.
virtual enum change_kind has_local_changes() const
friend var_diff_sptr compute_diff(const var_decl_sptr first, const var_decl_sptr second, diff_context_sptr ctxt)
Compute the diff between two instances of var_decl.
virtual const string & get_pretty_representation() const
diff_sptr type_diff() const
Getter for the diff of the types of the instances of var_decl.
var_decl_sptr second_var() const
Getter for the second var_decl of the diff.
The abstraction of an edit script for transforming a sequence A into a sequence B.
shared_ptr< subrange_type > subrange_sptr
Convenience typedef for a shared pointer on a function_decl::subrange.
shared_ptr< base_spec > base_spec_sptr
Convenience typedef.
vector< base_spec_sptr > base_specs
Convenience typedef.
vector< method_decl_sptr > member_functions
Convenience typedef.
bool get_is_anonymous() const
Test if the current declaration is anonymous.
The abstraction of the version of an ELF symbol.
Abstraction of an elf symbol.
const string & get_id_string() const
Get a string that is representative of a given elf_symbol.
std::vector< enumerator > enumerators
Convenience typedef for a list of enumerator.
Abstraction for a function declaration.
shared_ptr< parameter > parameter_sptr
Convenience typedef for a shared pointer on a function_decl::parameter.
const function_type_sptr get_type() const
Return the type of the current instance of function_decl.
interned_string get_id() const
Return an ID that tries to uniquely identify the function inside a program or a library.
An abstraction helper for type declarations.
The base class of both types and declarations.
change_kind
The kind of change the current function suppression should apply to.
@ ADDED_FUNCTION_CHANGE_KIND
The function was added to the second subject of the diff.
@ DELETED_FUNCTION_CHANGE_KIND
The function was deleted from the second subject of the diff.
change_kind
The kind of change the current variable suppression should apply to.
@ ADDED_VARIABLE_CHANGE_KIND
The variable was added to the second second subject of the diff.
@ DELETED_VARIABLE_CHANGE_KIND
The variable was deleted from the second subject of the diff.
bool has_void_ptr_to_ptr_change(const diff *dif)
Test if a diff node carries a void* to pointer type change.
diff_category has_fn_return_or_parm_harmful_change(const diff *d)
Test if a diff node is a function diff node that carries either a return or a parameter type change t...
diff_category has_var_harmful_local_change(const diff *d)
Test if a diff node carries a harmful local change to a variable.
std::vector< filter_base_sptr > filters
Convenience typedef for a vector of filter_base_sptr.
bool has_fn_with_virtual_offset_change(const diff *d)
Test if a diff node carries a change to the offset of a virtual function.
bool has_harmless_enum_to_int_change(const diff *diff)
Test if a diff node carries a harmless change of an enum into an integer (or vice-versa).
bool is_decl_only_class_with_size_change(const class_or_union &first, const class_or_union &second)
Test if two classes that are decl-only (have the decl-only flag and carry no data members) but are di...
shared_ptr< filter_base > filter_base_sptr
Convenience typedef for a shared pointer to filter_base.
bool has_benign_array_of_unknown_size_change(const diff *dif)
Test if a diff node carries a benign change to the size of a variable of type array.
bool has_harmful_name_change(const decl_base_sptr &f, const decl_base_sptr &s, const diff_context_sptr &ctxt)
Test if two decls represent a harmful name change.
bool has_decl_only_def_change(const decl_base_sptr &first, const decl_base_sptr &second)
Test if two decl_base_sptr are different just by the fact that one is decl-only and the other one is ...
bool has_basic_or_class_type_name_change(const diff *d)
Test if a diff node carries a basic or class type name change.
void apply_filter(filter_base &filter, corpus_diff_sptr d)
Walk the diff sub-trees of a a corpus_diff and apply a filter to the nodes visted....
bool is_mostly_distinct_diff(const diff *d)
Test if a diff node carries a distinct type change or a pointer/reference/typedef to distinct type ch...
bool is_harmful_category(diff_category c)
Test if an instance of diff_category (a category bit-field) is harmful or not.
shared_ptr< diff > diff_sptr
Convenience typedef for a shared_ptr for the diff class.
visiting_kind operator~(visiting_kind l)
The overloaded 'bit inversion' operator for visiting_kind.
const decl_diff_base * is_decl_diff(const diff *diff)
Test if a diff node is about differences between declarations.
const diff * peel_qualified_diff(const diff *dif)
If a diff node is about changes between two qualified types, get the diff node about changes between ...
const diff * peel_pointer_or_qualified_type_diff(const diff *dif)
If a diff node is about changes between two pointer, reference or qualified types,...
void categorize_redundancy(diff *diff_tree)
Walk a given diff sub-tree to categorize each of the nodes with respect to the REDUNDANT_CATEGORY.
vector< diff * > diff_ptrs_type
Convenience typedef for a vector of diff*.
shared_ptr< decl_diff_base > decl_diff_base_sptr
Convenience typedef for a shared_ptr of decl_diff_base.
void sort_string_diff_ptr_map(const string_diff_ptr_map &map, diff_ptrs_type &sorted)
Sort a map ofg string -> diff* into a vector of diff_ptr. The diff_ptr are sorted lexicographically w...
diff_category
An enum for the different categories that a diff tree node falls into, regarding the kind of changes ...
@ ACCESS_CHANGE_CATEGORY
This means the diff node (or at least one of its descendant nodes) carries access related changes,...
@ HARMLESS_DATA_MEMBER_CHANGE_CATEGORY
This means that a diff node in the sub-tree carries a harmless data member change....
@ SUPPRESSED_CATEGORY
This means that a diff node was marked as suppressed by a user-provided suppression specification.
@ VIRTUAL_MEMBER_CHANGE_CATEGORY
This means that a diff node in the sub-tree carries an incompatible change to a vtable.
@ REDUNDANT_CATEGORY
A diff node in this category is redundant. That means it's present as a child of a other nodes in the...
@ SIZE_OR_OFFSET_CHANGE_CATEGORY
This means the diff node (or at least one of its descendant nodes) carries a change that modifies the...
@ NON_VIRT_MEM_FUN_CHANGE_CATEGORY
This means that a diff node in the sub-tree carries an addition or removal of a non-virtual member fu...
@ HAS_DESCENDANT_WITH_ALLOWED_CHANGE_CATEGORY
A diff node in this category has a descendant node that is in the HAS_ALLOWED_CHANGE_CATEGORY categor...
@ HARMLESS_ENUM_CHANGE_CATEGORY
This means that a diff node in the sub-tree carries an addition of enumerator to an enum type.
@ FN_PARM_ADD_REMOVE_CHANGE_CATEGORY
A diff node in this category is a function (or function type) with at least one parameter added or re...
@ VOID_PTR_TO_PTR_CHANGE_CATEGORY
A diff node in this category carries a change from void pointer to non-void pointer.
@ NON_COMPATIBLE_DISTINCT_CHANGE_CATEGORY
A change between two non-compatible types of different kinds.
@ PRIVATE_TYPE_CATEGORY
This means that a diff node was warked as being for a private type. That is, the diff node is meant t...
@ NON_COMPATIBLE_NAME_CHANGE_CATEGORY
A non-compatible name change between two types.
@ COMPATIBLE_TYPE_CHANGE_CATEGORY
This means the diff node (or at least one of its descendant nodes) carries a change involving two com...
@ TYPE_DECL_ONLY_DEF_CHANGE_CATEGORY
This means that a diff node in the sub-tree carries a type that was declaration-only and that is now ...
@ STATIC_DATA_MEMBER_CHANGE_CATEGORY
This means that a diff node in the sub-tree carries an addition or removal of a static data member.
@ HARMLESS_UNION_OR_CLASS_CHANGE_CATEGORY
This means that a diff node in the sub-tree carries a harmless union or class change.
@ HARMLESS_DECL_NAME_CHANGE_CATEGORY
This means that a diff node in the sub-tree carries a harmless declaration name change....
@ NO_CHANGE_CATEGORY
This means the diff node does not carry any (meaningful) change, or that it carries changes that have...
@ HAS_PARENT_WITH_ALLOWED_CHANGE_CATEGORY
A diff node in this category has a parent node that is in the HAS_ALLOWED_CHANGE_CATEGORY category....
@ BENIGN_INFINITE_ARRAY_CHANGE_CATEGORY
A diff node in this category carries a change in the size of the array type of a global variable,...
@ VAR_TYPE_CV_CHANGE_CATEGORY
A diff node in this category is for a variable which type holds a cv-qualifier change.
@ HAS_ALLOWED_CHANGE_CATEGORY
A diff node in this category carries a change that must be reported, even if the diff node is also in...
@ FN_PARM_TYPE_CV_CHANGE_CATEGORY
A diff node in this category has a function parameter type with a cv-qualifiers change.
@ FN_PARM_TYPE_TOP_CV_CHANGE_CATEGORY
A diff node in this category is a function parameter type which top cv-qualifiers change.
@ FN_RETURN_TYPE_CV_CHANGE_CATEGORY
A diff node in this category is a function return type with a cv-qualifier change.
@ HARMLESS_SYMBOL_ALIAS_CHANGE_CATEGORY
This means that a diff node in the sub-tree carries an a symbol alias change that is harmless.
shared_ptr< reporter_base > reporter_base_sptr
A convenience typedef for a shared pointer to a reporter_base.
void sort_string_var_diff_sptr_map(const string_var_diff_sptr_map &map, var_diff_sptrs_type &sorted)
Sort of an instance of string_var_diff_sptr_map map.
unordered_map< string, var_diff_sptr > string_var_diff_sptr_map
Convenience typedef for a map whose key is a string and whose value is a changed variable of type var...
const class_or_union_diff * is_diff_of_class_or_union_type(const diff *d)
Test if a diff node represents a diff between two class or union types.
const subrange_diff * is_anonymous_subrange_diff(const diff *d)
Test if a diff node is a subrange_diff between two anonymous subranges.
const pointer_diff * is_pointer_diff(const diff *diff)
Test if a diff node is about differences between two pointers.
vector< var_diff_sptr > var_diff_sptrs_type
Convenience typedef for a vector of var_diff_sptr.
unordered_map< string, class_decl::base_spec_sptr > string_base_sptr_map
Convenience typedef for a map of string and class_decl::basse_spec_sptr.
void sort_string_function_ptr_map(const string_function_ptr_map &map, vector< const function_decl * > &sorted)
Sort an instance of string_function_ptr_map map and stuff a resulting sorted vector of pointers to fu...
diff_sptr try_to_diff(const type_or_decl_base_sptr first, const type_or_decl_base_sptr second, diff_context_sptr ctxt)
</distinct_diff>
bool has_basic_type_change_only(const diff *d)
Test if a diff node is a decl diff that only carries a basic type change on its type diff sub-node.
void propagate_categories(diff *diff_tree)
Visit all the nodes of a given sub-tree. For each node that has a particular category set,...
const subrange_diff * is_subrange_diff(const diff *diff)
Test if a diff node is a subrange_diff node.
unordered_map< unsigned, fn_parm_diff_sptr > unsigned_fn_parm_diff_sptr_map
Convenience typedef for a map which key is an integer and which value is a changed parameter.
void sort_string_base_sptr_map(const string_base_sptr_map &m, class_decl::base_specs &sorted)
Lexicographically sort base specifications found in instances of string_base_sptr_map.
diff_category get_default_harmless_categories_bitmap()
Getter of a bitmap made of the set of change categories that are considered harmless.
vector< diff_sptr > diff_sptrs_type
Convenience typedef for a vector of diff_sptr.
distinct_diff_sptr compute_diff_for_distinct_kinds(const type_or_decl_base_sptr first, const type_or_decl_base_sptr second, diff_context_sptr ctxt)
Try to diff entities that are of distinct kinds.
void sort_changed_data_members(changed_var_sptrs_type &input)
Sort (in place) a vector of changed data members.
unordered_map< string, type_base_sptr > string_type_base_sptr_map
Convenience typedef for a map which key is a string and which value is a type_base_sptr.
void print_category(diff_category c)
Print a given category out to stdout for debuging purposes.
diff_sptr try_to_diff< class_decl >(const type_or_decl_base_sptr first, const type_or_decl_base_sptr second, diff_context_sptr ctxt)
This is a specialization of try_to_diff() template to diff instances of class_decl.
shared_ptr< type_decl_diff > type_decl_diff_sptr
Convenience typedef for a shared pointer on a type_decl_diff type.
bool is_diff_of_variadic_parameter(const diff *d)
Test if a diff node represents the difference between a variadic parameter and something else.
shared_ptr< diff_context > diff_context_sptr
Convenience typedef for a shared pointer of diff_context.
bool is_diff_of_global_decls(const diff *)
Tests if a given diff node is to represent the changes between two gobal decls.
shared_ptr< subrange_diff > subrange_diff_sptr
A convenience typedef for a shared pointer to subrange_diff type.
bool is_diff_of_variadic_parameter_type(const diff *d)
Test if a diff node represents the difference between a variadic parameter type and something else.
diff_category get_default_harmful_categories_bitmap()
Getter of a bitmap made of the set of change categories that are considered harmful.
unordered_map< string, function_decl::parameter_sptr > string_parm_map
Convenience typedef for a map which value is a function parameter. The key is the name of the functio...
unordered_map< string, enum_type_decl::enumerator > string_enumerator_map
Convenience typedef for a map which value is an enumerator. The key is the name of the enumerator.
shared_ptr< var_diff > var_diff_sptr
Convenience typedef for a shared pointer to a var_diff type.
unordered_map< string, method_decl_sptr > string_member_function_sptr_map
Convenience typedef for a hash map of strings and member functions.
const diff * peel_typedef_diff(const diff *dif)
If a diff node is about changes between two typedef types, get the diff node about changes between th...
shared_ptr< reference_diff > reference_diff_sptr
Convenience typedef for a shared pointer on a reference_diff type.
vector< base_diff_sptr > base_diff_sptrs_type
Convenience typedef for a vector of base_diff_sptr.
shared_ptr< function_decl_diff > function_decl_diff_sptr
Convenience typedef for a shared pointer to a function_decl type.
shared_ptr< ptr_to_mbr_diff > ptr_to_mbr_diff_sptr
Typedef of a shared_ptr to ptr_to_mbr_diff.
void sort_string_elf_symbol_map(const string_elf_symbol_map &map, vector< elf_symbol_sptr > &sorted)
Sort a map of string -> pointer to elf_symbol.
const function_decl_diff * is_function_decl_diff(const diff *diff)
Test if a diff node is about differences between functions.
void sort_string_function_decl_diff_sptr_map(const string_function_decl_diff_sptr_map &map, function_decl_diff_sptrs_type &sorted)
Sort the values of a string_function_decl_diff_sptr_map map and store the result in a vector of funct...
shared_ptr< fn_parm_diff > fn_parm_diff_sptr
Convenience typedef for a shared pointer to a fn_parm_diff type.
unordered_map< string, function_decl_diff_sptr > string_function_decl_diff_sptr_map
Convenience typedef for a map which key is a string and which value is a function_decl_diff_sptr.
void clear_redundancy_categorization(diff *diff_tree)
Walk a given diff sub-tree to clear the REDUNDANT_CATEGORY out of the category of the nodes.
void sort_unsigned_data_member_diff_sptr_map(const unsigned_var_diff_sptr_map map, var_diff_sptrs_type &sorted)
Sort the values of a unsigned_var_diff_sptr_map map and store the result into a vector of var_diff_sp...
void sort_string_fn_parm_diff_sptr_map(const unsigned_fn_parm_diff_sptr_map &map, vector< fn_parm_diff_sptr > &sorted)
Sort a map of fn_parm_diff by the indexes of the function parameters.
bool is_less_than(const decl_diff_base &first, const decl_diff_base &second)
Compare two decl diff nodes (decl_diff_base) for the purpose of sorting.
bool is_child_node_of_base_diff(const diff *diff)
Test if a diff node is a child node of a base diff node.
visiting_kind
An enum for the different ways to visit a diff tree node.
@ SKIP_CHILDREN_VISITING_KIND
This says that the traversing code should avoid visiting the children nodes of the current node being...
@ DO_NOT_MARK_VISITED_NODES_AS_VISITED
This says that the traversing code should not mark visited nodes as having been traversed....
visiting_kind operator&(visiting_kind l, visiting_kind r)
The overloaded and operator for visiting_kind.
void sort_function_decl_diffs(function_decl_diff_sptrs_type &fn_diffs)
Sort a vector of function_decl_diff_sptr.
const diff * peel_fn_parm_diff(const diff *dif)
If a diff node is about changes between two function parameters get the diff node about changes betwe...
const function_type_diff * is_function_type_diff(const diff *diff)
Test if a diff node is a function_type_diff node.
const distinct_diff * is_distinct_diff(const diff *diff)
Test if a diff node is about differences between two diff nodes of different kinds.
unordered_map< string, changed_enumerator > string_changed_enumerator_map
Convenience typedef for a map which value is a changed enumerator. The key is the name of the changed...
visiting_kind operator|(visiting_kind l, visiting_kind r)
The overloaded or operator for visiting_kind.
const array_diff * is_array_diff(const diff *diff)
Test if a diff node is a array_diff node.
const diff * peel_reference_diff(const diff *dif)
If a diff node is about changes between two reference types, get the diff node about changes between ...
unordered_map< string, fn_parm_diff_sptr > string_fn_parm_diff_sptr_map
Convenience typedef for a map which value is a changed function parameter and which key is the name o...
string get_pretty_representation(diff *d)
Get a copy of the pretty representation of a diff node.
unordered_map< string, var_decl_sptr > string_var_ptr_map
Convenience typedef for a map which key is a string and which value is a point to var_decl.
void sort_artifacts_set(const artifact_sptr_set_type &set, vector< type_or_decl_base_sptr > &sorted)
Sort the set of ABI artifacts contained in a artifact_sptr_set_type.
const class_or_union_diff * is_anonymous_class_or_union_diff(const diff *d)
Test if a diff node is a class_or_union_diff between two anonymous classes or unions.
shared_ptr< base_diff > base_diff_sptr
Convenience typedef for a shared pointer to a base_diff type.
shared_ptr< scope_diff > scope_diff_sptr
Convenience typedef for a shared pointer on a scope_diff.
shared_ptr< class_diff > class_diff_sptr
Convenience typedef for a shared pointer on a class_diff type.
const base_diff * is_base_diff(const diff *diff)
Test if a diff node is about differences between two base class specifiers.
const fn_parm_diff * is_fn_parm_diff(const diff *diff)
Test if a diff node is about differences between two function parameters.
void sort_string_member_function_sptr_map(const string_member_function_sptr_map &map, class_or_union::member_functions &sorted)
Sort a map that's an instance of string_member_function_sptr_map and fill a vector of member function...
const union_diff * is_union_diff(const diff *diff)
Test if a diff node is a union_diff node.
vector< changed_var_sptr > changed_var_sptrs_type
Convenience typedef for a vector of @changed_var_sptr.gg381.
shared_ptr< pointer_diff > pointer_diff_sptr
Convenience typedef for a shared pointer on a pointer_diff type.
const diff * peel_pointer_diff(const diff *dif)
If a diff node is about changes between two pointer types, get the diff node about changes between th...
void sort_string_var_ptr_map(const string_var_ptr_map &map, vector< var_decl_sptr > &sorted)
Sort a map of string -> pointer to var_decl.
void apply_suppressions(diff *diff_tree)
Walk a given diff-sub tree and appply the suppressions carried by the context. If the suppression app...
void sort_string_parm_map(const string_parm_map &map, vector< function_decl::parameter_sptr > &sorted)
Sort a map of string -> function parameters.
void sort_string_type_base_sptr_map(string_type_base_sptr_map &map, vector< type_base_sptr > &sorted)
Sort a map of string to type_base_sptr entities.
const enum_diff * is_enum_diff(const diff *diff)
Test if a diff node is a enum_diff node.
unordered_map< string, base_diff_sptr > string_base_diff_sptr_map
Convenience typedef for a map of string and base_diff_sptr.
void sort_data_members(const string_decl_base_sptr_map &data_members, vector< decl_base_sptr > &sorted)
Sort a map of data members by the offset of their initial value.
const class_or_union_diff * is_class_or_union_diff(const diff *d)
Test if a diff node is a class_or_union_diff node.
const corpus_diff * is_corpus_diff(const diff *diff)
Test if a diff node is a corpus_diff node.
const diff * get_typedef_diff_underlying_type_diff(const diff *diff)
Return the leaf underlying diff node of a typedef_diff node.
const typedef_diff * is_typedef_diff(const diff *diff)
Test if a diff node is a typedef_diff node.
unordered_map< unsigned, var_diff_sptr > unsigned_var_diff_sptr_map
Convenience typedef for a map whose key is an unsigned int and whose value is a changed variable of t...
type_base_sptr get_leaf_type(qualified_type_def_sptr t)
Return the first underlying type that is not a qualified type.
const diff * peel_typedef_qualified_type_or_parameter_diff(const diff *dif)
If a diff node is about changes between two typedefs or qualified types, get the diff node about chan...
shared_ptr< array_diff > array_diff_sptr
Convenience typedef for a shared pointer on a array_diff type.
bool is_reference_or_ptr_diff_to_non_basic_nor_distinct_types(const diff *diff)
Test if a diff node is a reference or pointer diff node to a change that is neither basic type change...
const class_diff * is_class_diff(const diff *diff)
Test if a diff node is a class_diff node.
const type_decl_diff * is_diff_of_basic_type(const diff *d)
Test if a diff node represents a diff between two basic types.
unordered_map< string, diff_sptr > string_diff_sptr_map
Convenience typedef for a map which value is a diff_sptr. The key of the map is the qualified name of...
unordered_map< const diff *, artifact_sptr_set_type, diff_hash, diff_equal > diff_artifact_set_map_type
A convenience typedef for an unordered_map which key is a diff* and which value is a artifact_sptr_se...
const qualified_type_diff * is_qualified_type_diff(const diff *diff)
Test if a diff node is about differences between two qualified types.
diff_sptr compute_diff(const decl_base_sptr first, const decl_base_sptr second, diff_context_sptr ctxt)
Compute the difference between two decls. The decls can represent either type declarations,...
unordered_map< string, diff * > string_diff_ptr_map
Convenience typedef for a map which value is a diff*. The key of the map is the qualified name of the...
void sort_string_base_diff_sptr_map(const string_base_diff_sptr_map &map, base_diff_sptrs_type &sorted)
Sort a map of string -> base_diff_sptr into a sorted vector of base_diff_sptr. The base_diff_sptr are...
void sort_changed_enumerators(const string_changed_enumerator_map &enumerators_map, changed_enumerators_type &sorted)
Sort a map of changed enumerators.
void print_diff_tree(diff *diff_tree, ostream &out)
Emit a textual representation of a diff sub-tree to an output stream.
vector< changed_enumerator > changed_enumerators_type
Convenience typedef for a vector of changed enumerators.
vector< function_decl_diff_sptr > function_decl_diff_sptrs_type
Convenience typedef for a vector of function_decl_diff_sptr.
bool is_child_node_of_function_parm_diff(const diff *diff)
Test if a diff node is a child node of a function parameter diff node.
void sort_string_virtual_member_function_diff_sptr_map(const string_function_decl_diff_sptr_map &map, function_decl_diff_sptrs_type &sorted)
Sort an map of string -> virtual member function into a vector of virtual member functions....
shared_ptr< function_type_diff > function_type_diff_sptr
A convenience typedef for a shared pointer to function_type_type_diff.
const function_type_diff * is_function_type_diff_with_local_changes(const diff *diff)
Test if a given diff node carries a function type change with local changes.
const var_diff * is_var_diff(const diff *diff)
Test if a diff node is about differences between variables.
const type_diff_base * is_type_diff(const diff *diff)
Test if a diff node is about differences between types.
void sort_var_diffs(var_diff_sptrs_type &var_diffs)
Sort a vector of var_diff_sptr.
unordered_map< string, elf_symbol_sptr > string_elf_symbol_map
Convenience typedef for a map whose key is a string and whose value is an elf_symbol_sptr.
void sort_string_diff_sptr_map(const string_diff_sptr_map &map, diff_sptrs_type &sorted)
Sort a map ofg string -> diff_sptr into a vector of diff_sptr. The diff_sptr are sorted lexicographic...
const reference_diff * is_reference_diff(const diff *diff)
Test if a diff node is about differences between two references.
void sort_string_data_member_diff_sptr_map(const string_var_diff_sptr_map &map, var_diff_sptrs_type &sorted)
Sort the values of a string_var_diff_sptr_map and store the result in a vector of var_diff_sptr.
shared_ptr< distinct_diff > distinct_diff_sptr
Convenience typedef for a shared pointer to distinct_types_diff.
shared_ptr< corpus_diff > corpus_diff_sptr
A convenience typedef for a shared pointer to corpus_diff.
std::pair< var_decl_sptr, var_decl_sptr > changed_var_sptr
Convenience typedef for a pair of var_decl_sptr representing a var_decl change. The first member of t...
void apply_filters_and_categorize_diff_node_tree(diff_sptr &diff_tree)
Apply the diff tree filters that have been associated with the context of the a given diff,...
shared_ptr< typedef_diff > typedef_diff_sptr
Convenience typedef for a shared pointer on a typedef_diff type.
const diff * peel_typedef_or_qualified_type_diff(const diff *dif)
If a diff node is about changes between two typedefs or qualified types, get the diff node about chan...
void sort_enumerators(const string_enumerator_map &enumerators_map, enum_type_decl::enumerators &sorted)
Sort a map of enumerators by their value.
unordered_map< string, const function_decl * > string_function_ptr_map
Convenience typedef for a map which key is a string and which value is a pointer to decl_base.
ostream & operator<<(ostream &o, diff_category c)
Serialize an instance of diff_category to an output stream.
shared_ptr< translation_unit_diff > translation_unit_diff_sptr
Convenience typedef for a shared pointer on a translation_unit_diff type.
void compute_diff(RandomAccessOutputIterator a_base, RandomAccessOutputIterator a_begin, RandomAccessOutputIterator a_end, RandomAccessOutputIterator b_base, RandomAccessOutputIterator b_begin, RandomAccessOutputIterator b_end, vector< point > &lcs, edit_script &ses, int &ses_len)
Compute the longest common subsequence of two (sub-regions of) sequences as well as the shortest edit...
shared_ptr< reference_type_def > reference_type_def_sptr
Convenience typedef for a shared pointer on a reference_type_def.
bool get_member_function_is_dtor(const function_decl &f)
Test whether a member function is a destructor.
bool get_member_is_static(const decl_base &d)
Gets a flag saying if a class member is static or not.
shared_ptr< function_decl > function_decl_sptr
Convenience typedef for a shared pointer on a function_decl.
unordered_set< type_or_decl_base_sptr, type_or_decl_hash, type_or_decl_equal > artifact_sptr_set_type
A convenience typedef for a hash set of type_or_decl_base_sptr.
bool is_type(const type_or_decl_base &tod)
Test whether a declaration is a type.
bool is_anonymous_data_member(const decl_base &d)
Test if a decl is an anonymous data member.
array_type_def::subrange_type * is_subrange_type(const type_or_decl_base *type)
Test if a type is an array_type_def::subrange_type.
shared_ptr< elf_symbol > elf_symbol_sptr
A convenience typedef for a shared pointer to elf_symbol.
change_kind
A bitfield that gives callers of abigail::ir::equals() some insight about how different two internal ...
@ LOCAL_TYPE_CHANGE_KIND
This means that a given IR artifact has a local type change.
@ ALL_LOCAL_CHANGES_MASK
Testing (anding) against this mask means that a given IR artifact has local differences,...
@ LOCAL_NON_TYPE_CHANGE_KIND
This means that a given IR artifact has a local non-type change. That is a change that is carried by ...
bool is_user_defined_type(const type_base *t)
Test if a type is user-defined.
bool collect_non_anonymous_data_members(const class_or_union *cou, string_decl_base_sptr_map &dms)
Collect all the non-anonymous data members of a class or union type.
bool is_class_type(const type_or_decl_base &t)
Test whether a type is a class.
shared_ptr< array_type_def > array_type_def_sptr
Convenience typedef for a shared pointer on a array_type_def.
string get_pretty_representation(const type_or_decl_base *tod, bool internal)
Build and return a copy of the pretty representation of an ABI artifact that could be either a type o...
class_or_union * is_class_or_union_type(const type_or_decl_base *t)
Test if a type is a class_or_union.
const var_decl_sptr get_next_data_member(const class_or_union *klass, const var_decl_sptr &data_member)
In the context of a given class or union, this function returns the data member that is located after...
shared_ptr< class_decl > class_decl_sptr
Convenience typedef for a shared pointer on a class_decl.
shared_ptr< function_type > function_type_sptr
Convenience typedef for a shared pointer on a function_type.
shared_ptr< typedef_decl > typedef_decl_sptr
Convenience typedef for a shared pointer on a typedef_decl.
const enum_type_decl * is_enum_type(const type_or_decl_base *d)
Test if a decl is an enum_type_decl.
var_decl_sptr find_data_member_from_anonymous_data_member(const var_decl_sptr &anon_dm, const string &name)
Find a data member inside an anonymous data member.
const global_scope * get_global_scope(const decl_base &decl)
return the global scope as seen by a given declaration.
shared_ptr< var_decl > var_decl_sptr
Convenience typedef for a shared pointer on a var_decl.
shared_ptr< ptr_to_mbr_type > ptr_to_mbr_type_sptr
Convenience typedef for a shared pointer to a ptr_to_mbr_type.
shared_ptr< scope_decl > scope_decl_sptr
Convenience typedef for a shared pointer on a scope_decl.
shared_ptr< type_or_decl_base > type_or_decl_base_sptr
A convenience typedef for a shared_ptr to type_or_decl_base.
shared_ptr< translation_unit > translation_unit_sptr
Convenience typedef for a shared pointer on a translation_unit type.
unordered_map< string, decl_base_sptr > string_decl_base_sptr_map
Convenience typedef for a map which key is a string and which value is a decl_base_sptr.
bool equals(const decl_base &l, const decl_base &r, change_kind *k)
Compares two instances of decl_base.
shared_ptr< pointer_type_def > pointer_type_def_sptr
Convenience typedef for a shared pointer on a pointer_type_def.
bool is_enumerator_present_in_enum(const enum_type_decl::enumerator &enr, const enum_type_decl &enom)
Test if a given enumerator is found present in an enum.
uint64_t get_absolute_data_member_offset(const var_decl &m)
Get the absolute offset of a data member.
bool is_member_function(const function_decl &f)
Test whether a function_decl is a member function.
var_decl * is_var_decl(const type_or_decl_base *tod)
Tests if a declaration is a variable declaration.
decl_base * is_decl(const type_or_decl_base *d)
Test if an ABI artifact is a declaration.
shared_ptr< enum_type_decl > enum_type_decl_sptr
Convenience typedef for shared pointer to a enum_type_decl.
uint64_t get_data_member_offset(const var_decl &m)
Get the offset of a data member.
bool get_member_function_is_virtual(const function_decl &f)
Test if a given member function is virtual.
class_or_union * anonymous_data_member_to_class_or_union(const var_decl *d)
Get the class_or_union type of a given anonymous data member.
class_or_union * look_through_decl_only_class(class_or_union *the_class)
If a class (or union) is a decl-only class, get its definition. Otherwise, just return the initial cl...
bool is_union_type(const type_or_decl_base &t)
Test if a type is a union_decl.
bool is_data_member(const var_decl &v)
Test if a var_decl is a data member.
const decl_base * get_type_declaration(const type_base *t)
Get the declaration for a given type.
const var_decl * lookup_data_member(const type_base *type, const char *dm_name)
Look for a data member of a given class, struct or union type and return it.
interned_string get_function_id_or_pretty_representation(const function_decl *fn)
Get the ID of a function, or, if the ID can designate several different functions,...
shared_ptr< type_decl > type_decl_sptr
Convenience typedef for a shared pointer on a type_decl.
bool is_unique_type(const type_base_sptr &t)
Test if a type is unique in the entire environment.
bool types_are_compatible(const type_base_sptr type1, const type_base_sptr type2)
Test if two types are equal modulo a typedef or CV qualifiers.
function_decl * is_function_decl(const type_or_decl_base *d)
Test whether a declaration is a function_decl.
method_type_sptr is_method_type(const type_or_decl_base_sptr &t)
Test whether a type is a method_type.
bool class_or_union_types_of_same_kind(const class_or_union *first, const class_or_union *second)
Test if two class or union types are of the same kind.
type_base * peel_qualified_or_typedef_type(const type_base *type)
Return the leaf underlying type of a qualified or typedef type.
bool is_at_global_scope(const decl_base &decl)
Tests whether a given declaration is at global scope.
bool is_member_decl(const decl_base_sptr d)
Tests if a declaration is a class member.
shared_ptr< variable_suppression > variable_suppression_sptr
A convenience typedef for a shared pointer to variable_suppression.
vector< suppression_sptr > suppressions_type
Convenience typedef for a vector of suppression_sptr.
shared_ptr< function_suppression > function_suppression_sptr
Convenience typedef for a shared pointer to function_suppression.
variable_suppression_sptr is_variable_suppression(const suppression_sptr s)
Test if an instance of suppression is an instance of variable_suppression.
bool is_opaque_type_suppr_spec(const type_suppression &s)
Test if a type suppression specification represents a private type suppression automatically generate...
shared_ptr< type_suppression > type_suppression_sptr
Convenience typedef for a shared pointer to type_suppression.
function_suppression_sptr is_function_suppression(const suppression_sptr suppr)
Test if an instance of suppression is an instance of function_suppression.
type_suppression_sptr is_type_suppression(suppression_sptr suppr)
Test if an instance of suppression is an instance of type_suppression.
shared_ptr< suppression_base > suppression_sptr
Convenience typedef for a shared pointer to a suppression.
bool is_negated_suppression(const suppression_base &s)
Test if a suppression specification is a negated suppression.
Toplevel namespace for libabigail.
A comparison function for instances of base_diff.
A functor to compare instances of class_decl::base_spec.
A functor to compare two changed enumerators, based on their initial value.
size_t count_filtered_bases()
Count the number of bases classes whose changes got filtered out.
class_decl::base_spec_sptr base_has_changed(class_decl::base_spec_sptr) const
Test whether a given base class has changed. A base class has changed if it's in both in deleted *and...
The type of private data of class_or_union_diff.
size_t count_filtered_changed_dm(bool local_only=false)
Get the number of data member changes carried by the current diff node that were filtered out.
size_t count_filtered_subtype_changed_dm(bool local_only=false)
Get the number of data member sub-type changes carried by the current diff node that were filtered ou...
size_t get_deleted_non_static_data_members_number() const
Get the number of non static data members that were deleted.
size_t count_filtered_changed_mem_fns(const diff_context_sptr &)
Get the number of member functions changes carried by the current diff node that were filtered out.
decl_base_sptr subtype_changed_dm(decl_base_sptr) const
Test if the current diff node carries a data member change for a data member which name is the same a...
type_or_decl_base_sptr member_type_has_changed(decl_base_sptr) const
Test if the current diff node carries a member type change for a member type which name is the same a...
decl_base_sptr member_class_tmpl_has_changed(decl_base_sptr) const
Test if the current diff node carries a member class template change for a member class template whic...
size_t count_filtered_inserted_mem_fns(const diff_context_sptr &)
Get the number of member functions insertions carried by the current diff node that were filtered out...
size_t count_filtered_deleted_mem_fns(const diff_context_sptr &)
Get the number of member functions deletions carried by the current diff node that were filtered out.
size_t get_inserted_non_static_data_members_number() const
Get the number of non static data members that were inserted.
The type of the private data of corpus_diff::diff_stats.
bool added_unreachable_type_is_suppressed(const type_base *t) const
Test if an added type that is unreachable from public interface has been suppressed by a suppression ...
void ensure_lookup_tables_populated()
If the lookup tables are not yet built, walk the differences and fill the lookup tables.
const vector< diff_sptr > & changed_unreachable_types_sorted() const
Get the sorted vector of diff nodes representing changed unreachable types.
diff_context_sptr get_context()
Getter of the context associated with this corpus.
void categorize_redundant_changed_sub_nodes()
Walk the changed functions and variables diff nodes to categorize redundant nodes.
const string_diff_sptr_map & changed_unreachable_types() const
Get the map of diff nodes representing changed unreachable types.
bool added_unrefed_var_sym_is_suppressed(const elf_symbol *) const
Test if the change reports for a given added variable symbol (that is not referenced by any debug inf...
bool lookup_tables_empty() const
Tests if the lookup tables are empty.
bool deleted_unrefed_fn_sym_is_suppressed(const elf_symbol *) const
Test if the change reports for a given deleted function symbol (that is not referenced by any debug i...
void apply_filters_and_compute_diff_stats(corpus_diff::diff_stats &)
Compute the diff stats.
bool deleted_unreachable_type_is_suppressed(const type_base *t) const
Test if a deleted type that is unreachable from public interface has been suppressed by a suppression...
bool deleted_unrefed_var_sym_is_suppressed(const elf_symbol *) const
Test if the change reports for a given deleted variable symbol (that is not referenced by any debug i...
void maybe_dump_diff_tree()
If the user asked to dump the diff tree node (for changed variables and functions) on the error outpu...
void count_unreachable_types(size_t &num_added, size_t &num_removed, size_t &num_changed, size_t &num_filtered_added, size_t &num_filtered_removed, size_t &num_filtered_changed)
Count the number of types not reachable from the interface (i.e, not reachable from global functions ...
bool deleted_variable_is_suppressed(const var_decl_sptr &var) const
Test if the change reports for a give given deleted variable has been deleted.
void clear_redundancy_categorization()
Walk the changed functions and variables diff nodes and clear the redundancy categorization they migh...
void count_leaf_changes(size_t &num_changes, size_t &num_filtered)
Count the number of leaf changes as well as the number of the changes that have been filtered out.
void count_leaf_type_changes(size_t &num_type_changes, size_t &num_type_changes_filtered)
Count the number of leaf *type* changes as well as the number of the leaf type changes that have been...
bool added_variable_is_suppressed(const var_decl_sptr &var) const
Test if the change reports for a given added variable have been suppressed.
bool added_unrefed_fn_sym_is_suppressed(const elf_symbol *) const
Test if the change reports for a given added function symbol (that is not referenced by any debug inf...
bool deleted_function_is_suppressed(const function_decl *fn) const
Test if the change reports for a given deleted function have been deleted.
void apply_supprs_to_added_removed_fns_vars_unreachable_types()
Apply suppression specifications for this corpus diff to the set of added/removed functions/variables...
void emit_diff_stats(const diff_stats &stats, ostream &out, const string &indent)
Emit the summary of the functions & variables that got removed/changed/added.
bool added_function_is_suppressed(const function_decl *fn) const
Test if the change reports for a give given added function has been deleted.
void clear_lookup_tables()
Clear the lookup tables useful for reporting an enum_diff.
A comparison functor to compare two data members based on their offset.
A comparison functor to compare two instances of var_diff that represent changed data members based o...
A comparison functor for instances of diff.
A functor to compare two instances of diff_sptr.
The private data structure for distinct_diff.
A functor to compare instances of elf_symbol base on their names.
A functor to compare two enumerators based on their value. This implements the "less than" operator.
A comparison functor to compare two instances of fn_parm_diff based on their indexes.
"Less than" functor to compare instances of function_decl.
A "Less Than" functor to compare instance of function_decl_diff.
Functor that compares two function parameters for the purpose of sorting them.
The internal type for the impl idiom implementation of pointer_diff.
The private data of the ptr_to_mbr_diff type.
The internal type for the impl idiom implementation of subrange_diff.
A functor to compare instances of var_decl base on their qualified names.
The internal type for the impl idiom implementation of var_diff.
Functor to sort instances of var_diff_sptr.
A comparison functor for instances of function_decl_diff that represent changes between two virtual m...
An equality functor to deeply compare pointers.
A comparison functor to compare pointer to instances of type_or_decl_base.
A deleter for shared pointers that ... doesn't delete the object managed by the shared pointer.