1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
// -*- Mode: C++ -*-
//
// Copyright (C) 2013-2025 Red Hat, Inc.
///@file
#ifndef __ABG_TOOLS_UTILS_H
#define __ABG_TOOLS_UTILS_H
#include <iostream>
#include <istream>
#include <memory>
#include <ostream>
#include <set>
#include <string>
#include "abg-suppression.h"
#include "abg-elf-based-reader.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;
void initialize();
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();
const char* get_anonymous_subrange_internal_name_prefix();
bool file_exists(const string&);
bool is_regular_file(const string&);
bool file_has_dwarf_debug_info(const string& elf_file_path,
const vector<string>& debug_info_root_paths);
bool file_has_ctf_debug_info(const string& elf_file_path,
const vector<string>& debug_info_root_paths);
bool file_has_btf_debug_info(const string& elf_file_path,
const vector<string>& debug_info_root_paths);
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<string>&);
bool string_suffix(const string&, const string&, string&);
bool sorted_strings_common_prefix(vector<string>&, string&);
string get_library_version_string();
string get_abixml_version_string();
bool execute_command_and_get_output(const string&, vector<string>&);
void get_comma_separated_args_of_option(const string& input_str,
const string& option,
vector<string>& arguments);
bool get_dsos_provided_by_rpm(const string& rpm_path,
set<string>& provided_dsos);
string trim_white_space(const string&);
bool remove_white_spaces(string&);
bool normalize_litterals(string&);
string trim_leading_string(const string& from, const string& to_trim);
void convert_char_stars_to_char_star_stars(const vector<char*>&,
vector<char**>&);
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<string>& hdr_files);
suppr::type_suppression_sptr
gen_suppr_spec_from_headers(const vector<string>& headers_root_dirs,
const vector<string>& header_files);
suppr::suppressions_type
gen_suppr_spec_from_kernel_abi_whitelists
(const vector<string>& abi_whitelist_paths);
bool
get_file_path_dirs_under_dir(const string& root_dir, vector<string>& dirs);
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<string>& module_paths);
bool
get_binary_paths_from_kernel_dist(const string& dist_root,
string& vmlinux_path,
vector<string>& module_paths);
bool
get_file_path_dirs_under_dir(const string& root, vector<string>& dirs);
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);
bool
find_file_under_dirs(const vector<string>& root_dirs,
const string& file_path_to_look_for,
string& result);
bool
get_dependencies(const corpus&, const vector<string>&, set<string>&);
void
add_binaries_into_corpus_group(const fe_iface_sptr& reader,
const vector<string>& binaries,
const vector<string>& deps_dirs,
corpus_group& group);
void
add_dependencies_into_corpus_group(const fe_iface_sptr& reader,
const corpus& korpus,
const vector<string>& deps_dirs,
corpus_group& group);
corpus_group_sptr
stick_corpus_and_binaries_into_corpus_group(const fe_iface_sptr& reader,
const corpus_sptr& korpus,
const vector<string>& binaries,
const vector<string>& deps_dirs);
corpus_group_sptr
stick_corpus_and_dependencies_into_corpus_group(const fe_iface_sptr& reader,
const corpus_sptr& korpus,
const vector<string>& deps_dirs);
class temp_file;
/// Convenience typedef for a shared_ptr to @ref temp_file.
typedef shared_ptr<temp_file> 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;
std::unique_ptr<priv> 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,
// All non-tared compression scheme go under here. When one of
// these is returned, the goal is to look into the uncompressed
// stream to get what format has been compressed, then return an
// enumerator for that compressed format instead.
//
// Please note that each time a new enumerator is added here, one
// needs to add a corresponding enumerator to the @ref
// compression_kind enum in abg-tools-utils.cc and update the
// is_compressed_file_type and get_compressed_streambuf functions
// accordingly.
/// The XZ (lzma) compresson scheme.
FILE_TYPE_XZ
};
/// 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;
std::unique_ptr<priv> 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 look_through_compression = true);
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
rpm_contains_file(const string& rpm_path,
const string& file_name);
bool
file_is_kernel_debuginfo_package(const string& file_path,
file_type file_type);
std::shared_ptr<char>
make_path_absolute(const char*p);
string
make_path_absolute(const string& p);
corpus_group_sptr
build_corpus_group_from_kernel_dist_under(const string& root,
const string debug_info_root,
const string& vmlinux_path,
vector<string>& suppr_paths,
vector<string>& kabi_wl_paths,
suppr::suppressions_type& supprs,
bool verbose,
environment& env,
corpus::origin requested_fe_kind = corpus::DWARF_ORIGIN);
elf_based_reader_sptr
create_best_elf_based_reader(const string& elf_file_path,
const vector<string>& debug_info_root_paths,
environment& env,
corpus::origin requested_debug_info_kind,
bool show_all_types,
bool linux_kernel_mode = false);
/// This is a custom std::streambuf that knows how to decompress an
/// input stream that was compressed using xz.
///
/// The code was inspired by the example in the source code of the xz
/// project at
/// https://github.com/tukaani-project/xz/blob/master/doc/examples/02_decompress.c.
///
/// here is an example of how a user code would use this custom
/// streambuf to decode an xz'ed file and emit its content to stdout.
///
/// ifstream input_file("/path/to/a/compressed/file.xz", ifstream::binary);
/// xz_decompressor_type xzed_streambuf(input_file);
/// istream input_stream(&xzed_streambuf);
///
/// const size_t BUFFER_SIZE = 1024 * 4;
/// vector<char> decompressed_data(BUFFER_SIZE);
/// input_stream.read(decompressed_data.data(), BUFFER_SIZE);
/// size_t nb_bytes_read = input_stream.gcount();
/// while (nb_bytes_read && !input_stream.bad())
/// {
/// for (auto c : decompressed_data)
/// std::out << c;
/// input_stream.read(decompressed_data.data(), BUFFER_SIZE);
/// nb_bytes_read = input_stream.gcount();
/// }
/// input_file.close();
///
/// Voila.
class xz_decompressor_type : public std::streambuf
{
struct priv;
std::unique_ptr<priv> priv_;
public:
xz_decompressor_type(std::istream& xz_istream);
~xz_decompressor_type();
protected:
int_type
underflow() override;
}; // end class xz_decompressor_type.
}// 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
|