/* SPDX-License-Identifier: GPL-2.0-or-later */ /* * Copyright (C) 2015 Josh Poimboeuf <jpoimboe@redhat.com> */ #ifndef _WARN_H #define _WARN_H #include <stdlib.h> #include <string.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <objtool/builtin.h> #include <objtool/elf.h> extern const char *objname; static inline char *offstr(struct section *sec, unsigned long offset) { bool is_text = (sec->sh.sh_flags & SHF_EXECINSTR); struct symbol *sym = NULL; char *str; int len; if (is_text) sym = find_func_containing(sec, offset); if (!sym) sym = find_symbol_containing(sec, offset); if (sym) { str = malloc(strlen(sym->name) + strlen(sec->name) + 40); len = sprintf(str, "%s+0x%lx", sym->name, offset - sym->offset); if (opts.sec_address) sprintf(str+len, " (%s+0x%lx)", sec->name, offset); } else { str = malloc(strlen(sec->name) + 20); sprintf(str, "%s+0x%lx", sec->name, offset); } return str; } #define WARN(format, ...) \ fprintf(stderr, \ "%s: warning: objtool: " format "\n", \ objname, ##__VA_ARGS__) #define WARN_FUNC(format, sec, offset, ...) \ ({ \ char *_str = offstr(sec, offset); \ WARN("%s: " format, _str, ##__VA_ARGS__); \ free(_str); \ }) #define WARN_INSN(insn, format, ...) \ ({ \ struct instruction *_insn = (insn); \ if (!_insn->sym || !_insn->sym->warned) \ WARN_FUNC(format, _insn->sec, _insn->offset, \ ##__VA_ARGS__); \ if (_insn->sym) \ _insn->sym->warned = 1; \ }) #define BT_INSN(insn, format, ...) \ ({ \ if (opts.verbose || opts.backtrace) { \ struct instruction *_insn = (insn); \ char *_str = offstr(_insn->sec, _insn->offset); \ WARN(" %s: " format, _str, ##__VA_ARGS__); \ free(_str); \ } \ }) #define WARN_ELF(format, ...) \ WARN(format ": %s", ##__VA_ARGS__, elf_errmsg(-1)) #endif /* _WARN_H */