Merge tag 'trace-printf-v6.13' of git://git.kernel.org/pub/scm/linux/kernel/git/trace...
[drm/drm-misc.git] / tools / perf / util / llvm-c-helpers.cpp
blob663bcaba2041fc256266abd0c008832a3f525a7e
1 // SPDX-License-Identifier: GPL-2.0
3 /*
4 * Must come before the linux/compiler.h include, which defines several
5 * macros (e.g. noinline) that conflict with compiler builtins used
6 * by LLVM.
7 */
8 #pragma GCC diagnostic push
9 #pragma GCC diagnostic ignored "-Wunused-parameter" /* Needed for LLVM <= 15 */
10 #include <llvm/DebugInfo/Symbolize/Symbolize.h>
11 #include <llvm/Support/TargetSelect.h>
12 #pragma GCC diagnostic pop
14 #include <inttypes.h>
15 #include <stdio.h>
16 #include <sys/types.h>
17 #include <linux/compiler.h>
18 extern "C" {
19 #include <linux/zalloc.h>
21 #include "symbol_conf.h"
22 #include "llvm-c-helpers.h"
24 extern "C"
25 char *dso__demangle_sym(struct dso *dso, int kmodule, const char *elf_name);
27 using namespace llvm;
28 using llvm::symbolize::LLVMSymbolizer;
31 * Allocate a static LLVMSymbolizer, which will live to the end of the program.
32 * Unlike the bfd paths, LLVMSymbolizer has its own cache, so we do not need
33 * to store anything in the dso struct.
35 static LLVMSymbolizer *get_symbolizer()
37 static LLVMSymbolizer *instance = nullptr;
38 if (instance == nullptr) {
39 LLVMSymbolizer::Options opts;
41 * LLVM sometimes demangles slightly different from the rest
42 * of the code, and this mismatch can cause new_inline_sym()
43 * to get confused and mark non-inline symbol as inlined
44 * (since the name does not properly match up with base_sym).
45 * Thus, disable the demangling and let the rest of the code
46 * handle it.
48 opts.Demangle = false;
49 instance = new LLVMSymbolizer(opts);
51 return instance;
54 /* Returns 0 on error, 1 on success. */
55 static int extract_file_and_line(const DILineInfo &line_info, char **file,
56 unsigned int *line)
58 if (file) {
59 if (line_info.FileName == "<invalid>") {
60 /* Match the convention of libbfd. */
61 *file = nullptr;
62 } else {
63 /* The caller expects to get something it can free(). */
64 *file = strdup(line_info.FileName.c_str());
65 if (*file == nullptr)
66 return 0;
69 if (line)
70 *line = line_info.Line;
71 return 1;
74 extern "C"
75 int llvm_addr2line(const char *dso_name, u64 addr,
76 char **file, unsigned int *line,
77 bool unwind_inlines,
78 llvm_a2l_frame **inline_frames)
80 LLVMSymbolizer *symbolizer = get_symbolizer();
81 object::SectionedAddress sectioned_addr = {
82 addr,
83 object::SectionedAddress::UndefSection
86 if (unwind_inlines) {
87 Expected<DIInliningInfo> res_or_err =
88 symbolizer->symbolizeInlinedCode(dso_name,
89 sectioned_addr);
90 if (!res_or_err)
91 return 0;
92 unsigned num_frames = res_or_err->getNumberOfFrames();
93 if (num_frames == 0)
94 return 0;
96 if (extract_file_and_line(res_or_err->getFrame(0),
97 file, line) == 0)
98 return 0;
100 *inline_frames = (llvm_a2l_frame *)calloc(
101 num_frames, sizeof(**inline_frames));
102 if (*inline_frames == nullptr)
103 return 0;
105 for (unsigned i = 0; i < num_frames; ++i) {
106 const DILineInfo &src = res_or_err->getFrame(i);
108 llvm_a2l_frame &dst = (*inline_frames)[i];
109 if (src.FileName == "<invalid>")
110 /* Match the convention of libbfd. */
111 dst.filename = nullptr;
112 else
113 dst.filename = strdup(src.FileName.c_str());
114 dst.funcname = strdup(src.FunctionName.c_str());
115 dst.line = src.Line;
117 if (dst.filename == nullptr ||
118 dst.funcname == nullptr) {
119 for (unsigned j = 0; j <= i; ++j) {
120 zfree(&(*inline_frames)[j].filename);
121 zfree(&(*inline_frames)[j].funcname);
123 zfree(inline_frames);
124 return 0;
128 return num_frames;
129 } else {
130 if (inline_frames)
131 *inline_frames = nullptr;
133 Expected<DILineInfo> res_or_err =
134 symbolizer->symbolizeCode(dso_name, sectioned_addr);
135 if (!res_or_err)
136 return 0;
137 return extract_file_and_line(*res_or_err, file, line);
141 static char *
142 make_symbol_relative_string(struct dso *dso, const char *sym_name,
143 u64 addr, u64 base_addr)
145 if (!strcmp(sym_name, "<invalid>"))
146 return NULL;
148 char *demangled = dso__demangle_sym(dso, 0, sym_name);
149 if (base_addr && base_addr != addr) {
150 char buf[256];
151 snprintf(buf, sizeof(buf), "%s+0x%" PRIx64,
152 demangled ? demangled : sym_name, addr - base_addr);
153 free(demangled);
154 return strdup(buf);
155 } else {
156 if (demangled)
157 return demangled;
158 else
159 return strdup(sym_name);
163 extern "C"
164 char *llvm_name_for_code(struct dso *dso, const char *dso_name, u64 addr)
166 LLVMSymbolizer *symbolizer = get_symbolizer();
167 object::SectionedAddress sectioned_addr = {
168 addr,
169 object::SectionedAddress::UndefSection
171 Expected<DILineInfo> res_or_err =
172 symbolizer->symbolizeCode(dso_name, sectioned_addr);
173 if (!res_or_err) {
174 return NULL;
176 return make_symbol_relative_string(
177 dso, res_or_err->FunctionName.c_str(),
178 addr, res_or_err->StartAddress ? *res_or_err->StartAddress : 0);
181 extern "C"
182 char *llvm_name_for_data(struct dso *dso, const char *dso_name, u64 addr)
184 LLVMSymbolizer *symbolizer = get_symbolizer();
185 object::SectionedAddress sectioned_addr = {
186 addr,
187 object::SectionedAddress::UndefSection
189 Expected<DIGlobal> res_or_err =
190 symbolizer->symbolizeData(dso_name, sectioned_addr);
191 if (!res_or_err) {
192 return NULL;
194 return make_symbol_relative_string(
195 dso, res_or_err->Name.c_str(),
196 addr, res_or_err->Start);