// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // -*- Mode: C++ -*- // // Copyright (C) 2013-2021 Red Hat, Inc. ///@file #ifndef __ABG_TOOLS_UTILS_H #define __ABG_TOOLS_UTILS_H #include #include #include #include #include #include #include "abg-suppression.h" namespace abigail { namespace tools_utils { using std::ostream; using std::istream; using std::ifstream; using std::string; using std::set; using std::shared_ptr; const char* get_system_libdir(); const char* get_anonymous_struct_internal_name_prefix(); const char* get_anonymous_union_internal_name_prefix(); const char* get_anonymous_enum_internal_name_prefix(); bool file_exists(const string&); bool is_regular_file(const string&); bool is_dir(const string&); bool dir_exists(const string&); bool dir_is_empty(const string &); bool decl_names_equal(const string&, const string&); bool maybe_get_symlink_target_file_path(const string& file_path, string& target_path); bool base_name(string const& path, string& file_name); bool dir_name(string const &path, string& path_dir_name, bool keep_separator_at_end=false); void real_path(const string&path, string& realpath); bool ensure_dir_path_created(const string&); bool ensure_parent_dir_created(const string&); ostream& emit_prefix(const string& prog_name, ostream& out); bool check_file(const string& path, ostream& out, const string& prog_name = ""); bool check_dir(const string& path, ostream& out, const string& prog_name=""); bool string_ends_with(const string&, const string&); bool string_begins_with(const string&, const string&); bool string_is_ascii(const string&); bool string_is_ascii_identifier(const string&); bool split_string(const string&, const string&, vector&); bool string_suffix(const string&, const string&, string&); bool sorted_strings_common_prefix(vector&, string&); string get_library_version_string(); bool execute_command_and_get_output(const string&, vector&); bool get_dsos_provided_by_rpm(const string& rpm_path, set& provided_dsos); string trim_white_space(const string&); string trim_leading_string(const string& from, const string& to_trim); void convert_char_stars_to_char_star_stars(const vector&, vector&); suppr::type_suppression_sptr gen_suppr_spec_from_headers(const string& hdrs_root_dir); suppr::type_suppression_sptr gen_suppr_spec_from_headers(const string& hdrs_root_dir, const vector& hdr_files); suppr::type_suppression_sptr gen_suppr_spec_from_headers(const vector& headers_root_dirs, const vector& header_files); suppr::suppressions_type gen_suppr_spec_from_kernel_abi_whitelists (const vector& abi_whitelist_paths); bool get_vmlinux_path_from_kernel_dist(const string& from, string& vmlinux_path); bool get_binary_paths_from_kernel_dist(const string& dist_root, const string& debug_info_root_path, string& vmlinux_path, vector& module_paths); bool get_binary_paths_from_kernel_dist(const string& dist_root, string& vmlinux_path, vector& module_paths); string get_default_system_suppression_file_path(); string get_default_user_suppression_file_path(); void load_default_system_suppressions(suppr::suppressions_type&); void load_default_user_suppressions(suppr::suppressions_type&); bool find_file_under_dir(const string& root_dir, const string& file_path_to_look_for, string& result); class temp_file; /// Convenience typedef for a shared_ptr to @ref temp_file. typedef shared_ptr temp_file_sptr; /// A temporary file. /// /// This is a helper file around the mkstemp API. /// /// Once the temporary file is created, users can interact with it /// using an fstream. They can also get the path to the newly /// created temporary file. /// /// When the instance of @ref temp_file is destroyed, the underlying /// resources are de-allocated, the underlying temporary file is /// closed and removed. class temp_file { struct priv; typedef shared_ptr priv_sptr; priv_sptr priv_; temp_file(); public: bool is_good() const; const char* get_path() const; std::fstream& get_stream(); static temp_file_sptr create(); }; // end class temp_file size_t get_random_number(); string get_random_number_as_string(); /// The different types of files understood the bi* suite of tools. enum file_type { /// A file type we don't know about. FILE_TYPE_UNKNOWN, /// The native xml file format representing a translation unit. FILE_TYPE_NATIVE_BI, /// An elf file. Read this kind of file should yield an /// abigail::corpus type. FILE_TYPE_ELF, /// An archive (AR) file. FILE_TYPE_AR, // A native abixml file format representing a corpus of one or // several translation units. FILE_TYPE_XML_CORPUS, // A native abixml file format representing a corpus group of one or // several corpora. FILE_TYPE_XML_CORPUS_GROUP, /// An RPM (.rpm) binary file FILE_TYPE_RPM, /// An SRPM (.src.rpm) file FILE_TYPE_SRPM, /// A DEB (.deb) binary file FILE_TYPE_DEB, /// A plain directory FILE_TYPE_DIR, /// A tar archive. The archive can be compressed with the popular /// compression schemes recognized by GNU tar. FILE_TYPE_TAR }; /// Exit status for abidiff and abicompat tools. /// /// It's actually a bit mask. The value of each enumerator is a power /// of two. enum abidiff_status { /// This is for when the compared ABIs are equal. /// /// Its numerical value is 0. ABIDIFF_OK = 0, /// This bit is set if there is an application error. /// /// Its numerical value is 1. ABIDIFF_ERROR = 1, /// This bit is set if the tool is invoked in an non appropriate /// manner. /// /// Its numerical value is 2. ABIDIFF_USAGE_ERROR = 1 << 1, /// This bit is set if the ABIs being compared are different. /// /// Its numerical value is 4. ABIDIFF_ABI_CHANGE = 1 << 2, /// This bit is set if the ABIs being compared are different *and* /// are incompatible. /// /// Its numerical value is 8. ABIDIFF_ABI_INCOMPATIBLE_CHANGE = 1 << 3 }; abidiff_status operator|(abidiff_status, abidiff_status); abidiff_status operator&(abidiff_status, abidiff_status); abidiff_status& operator|=(abidiff_status&l, abidiff_status r); bool abidiff_status_has_error(abidiff_status s); bool abidiff_status_has_abi_change(abidiff_status s); bool abidiff_status_has_incompatible_abi_change(abidiff_status s); /// A type used to time various part of the libabigail system. class timer { struct priv; typedef shared_ptr priv_sptr; priv_sptr priv_; public: enum kind { /// Default timer kind. DEFAULT_TIMER_KIND = 0, /// This kind of timer starts upon instantiation. START_ON_INSTANTIATION_TIMER_KIND = 1, }; timer (kind k = DEFAULT_TIMER_KIND); bool start(); bool stop(); time_t value_in_seconds() const; bool value(time_t& hours, time_t& minutes, time_t& seconds, time_t& milliseconds) const; string value_as_string() const; ~timer(); }; //end class timer ostream& operator<<(ostream&, const timer&); ostream& operator<<(ostream& output, file_type r); file_type guess_file_type(istream& in); file_type guess_file_type(const string& file_path); bool get_rpm_name(const string& str, string& name); bool get_rpm_arch(const string& str, string& arch); bool get_deb_name(const string& str, string& name); bool file_is_kernel_package(const string& file_path, file_type file_type); bool file_is_kernel_debuginfo_package(const string& file_path, file_type file_type); std::shared_ptr make_path_absolute(const char*p); char* make_path_absolute_to_be_freed(const char*p); corpus_group_sptr build_corpus_group_from_kernel_dist_under(const string& root, const string debug_info_root, const string& vmlinux_path, vector& suppr_paths, vector& kabi_wl_paths, suppr::suppressions_type& supprs, bool verbose, environment_sptr& env); }// end namespace tools_utils /// A macro that expands to aborting the program when executed. /// /// Before aborting, the macro emits informatin about the source /// location where it was expanded. #define ABG_ASSERT_NOT_REACHED \ do { \ std::cerr << "in " << __FUNCTION__ \ << " at: " << __FILE__ << ":" << __LINE__ \ << ": execution should not have reached this point!\n"; \ abort(); \ } while (false) }//end namespace abigail #endif //__ABG_TOOLS_UTILS_H