Include backtrace in verbose mode. This makes it easy to gather all the information needed for diagnosing objtool warnings. Reviewed-by: Miroslav Benes <mbenes@suse.cz> Link: https://lore.kernel.org/r/c255224fabcf7e64bac232fec1c77c9fc2d7d7ab.1681853186.git.jpoimboe@kernel.org Signed-off-by: Josh Poimboeuf <jpoimboe@kernel.org>
79 lines
1.9 KiB
C
79 lines
1.9 KiB
C
/* 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 */
|