1 /* SPDX-License-Identifier: GPL-2.0-or-later */
2 /*
3  * Copyright (C) 2015 Josh Poimboeuf <jpoimboe@redhat.com>
4  */
5 
6 #ifndef _WARN_H
7 #define _WARN_H
8 
9 #include <stdlib.h>
10 #include <string.h>
11 #include <sys/types.h>
12 #include <sys/stat.h>
13 #include <fcntl.h>
14 #include <objtool/builtin.h>
15 #include <objtool/elf.h>
16 
17 extern const char *objname;
18 
offstr(struct section * sec,unsigned long offset)19 static inline char *offstr(struct section *sec, unsigned long offset)
20 {
21 	bool is_text = (sec->sh.sh_flags & SHF_EXECINSTR);
22 	struct symbol *sym = NULL;
23 	char *str;
24 	int len;
25 
26 	if (is_text)
27 		sym = find_func_containing(sec, offset);
28 	if (!sym)
29 		sym = find_symbol_containing(sec, offset);
30 
31 	if (sym) {
32 		str = malloc(strlen(sym->name) + strlen(sec->name) + 40);
33 		len = sprintf(str, "%s+0x%lx", sym->name, offset - sym->offset);
34 		if (opts.sec_address)
35 			sprintf(str+len, " (%s+0x%lx)", sec->name, offset);
36 	} else {
37 		str = malloc(strlen(sec->name) + 20);
38 		sprintf(str, "%s+0x%lx", sec->name, offset);
39 	}
40 
41 	return str;
42 }
43 
44 #define WARN(format, ...)				\
45 	fprintf(stderr,					\
46 		"%s: warning: objtool: " format "\n",	\
47 		objname, ##__VA_ARGS__)
48 
49 #define WARN_FUNC(format, sec, offset, ...)		\
50 ({							\
51 	char *_str = offstr(sec, offset);		\
52 	WARN("%s: " format, _str, ##__VA_ARGS__);	\
53 	free(_str);					\
54 })
55 
56 #define WARN_INSN(insn, format, ...)					\
57 ({									\
58 	struct instruction *_insn = (insn);				\
59 	if (!_insn->sym || !_insn->sym->warned)				\
60 		WARN_FUNC(format, _insn->sec, _insn->offset,		\
61 			  ##__VA_ARGS__);				\
62 	if (_insn->sym)							\
63 		_insn->sym->warned = 1;					\
64 })
65 
66 #define BT_INSN(insn, format, ...)				\
67 ({								\
68 	if (opts.verbose || opts.backtrace) {			\
69 		struct instruction *_insn = (insn);		\
70 		char *_str = offstr(_insn->sec, _insn->offset); \
71 		WARN("  %s: " format, _str, ##__VA_ARGS__);	\
72 		free(_str);					\
73 	}							\
74 })
75 
76 #define WARN_ELF(format, ...)				\
77 	WARN(format ": %s", ##__VA_ARGS__, elf_errmsg(-1))
78 
79 #endif /* _WARN_H */
80