~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

TOMOYO Linux Cross Reference
Linux/tools/perf/util/llvm-c-helpers.cpp

Version: ~ [ linux-6.12-rc7 ] ~ [ linux-6.11.7 ] ~ [ linux-6.10.14 ] ~ [ linux-6.9.12 ] ~ [ linux-6.8.12 ] ~ [ linux-6.7.12 ] ~ [ linux-6.6.60 ] ~ [ linux-6.5.13 ] ~ [ linux-6.4.16 ] ~ [ linux-6.3.13 ] ~ [ linux-6.2.16 ] ~ [ linux-6.1.116 ] ~ [ linux-6.0.19 ] ~ [ linux-5.19.17 ] ~ [ linux-5.18.19 ] ~ [ linux-5.17.15 ] ~ [ linux-5.16.20 ] ~ [ linux-5.15.171 ] ~ [ linux-5.14.21 ] ~ [ linux-5.13.19 ] ~ [ linux-5.12.19 ] ~ [ linux-5.11.22 ] ~ [ linux-5.10.229 ] ~ [ linux-5.9.16 ] ~ [ linux-5.8.18 ] ~ [ linux-5.7.19 ] ~ [ linux-5.6.19 ] ~ [ linux-5.5.19 ] ~ [ linux-5.4.285 ] ~ [ linux-5.3.18 ] ~ [ linux-5.2.21 ] ~ [ linux-5.1.21 ] ~ [ linux-5.0.21 ] ~ [ linux-4.20.17 ] ~ [ linux-4.19.323 ] ~ [ linux-4.18.20 ] ~ [ linux-4.17.19 ] ~ [ linux-4.16.18 ] ~ [ linux-4.15.18 ] ~ [ linux-4.14.336 ] ~ [ linux-4.13.16 ] ~ [ linux-4.12.14 ] ~ [ linux-4.11.12 ] ~ [ linux-4.10.17 ] ~ [ linux-4.9.337 ] ~ [ linux-4.4.302 ] ~ [ linux-3.10.108 ] ~ [ linux-2.6.32.71 ] ~ [ linux-2.6.0 ] ~ [ linux-2.4.37.11 ] ~ [ unix-v6-master ] ~ [ ccs-tools-1.8.12 ] ~ [ policy-sample ] ~
Architecture: ~ [ i386 ] ~ [ alpha ] ~ [ m68k ] ~ [ mips ] ~ [ ppc ] ~ [ sparc ] ~ [ sparc64 ] ~

Diff markup

Differences between /tools/perf/util/llvm-c-helpers.cpp (Architecture alpha) and /tools/perf/util/llvm-c-helpers.cpp (Architecture ppc)


  1 // SPDX-License-Identifier: GPL-2.0                 1 // SPDX-License-Identifier: GPL-2.0
  2                                                     2 
  3 /*                                                  3 /*
  4  * Must come before the linux/compiler.h inclu      4  * Must come before the linux/compiler.h include, which defines several
  5  * macros (e.g. noinline) that conflict with c      5  * macros (e.g. noinline) that conflict with compiler builtins used
  6  * by LLVM.                                         6  * by LLVM.
  7  */                                                 7  */
  8 #pragma GCC diagnostic push                         8 #pragma GCC diagnostic push
  9 #pragma GCC diagnostic ignored "-Wunused-param      9 #pragma GCC diagnostic ignored "-Wunused-parameter"  /* Needed for LLVM <= 15 */
 10 #include <llvm/DebugInfo/Symbolize/Symbolize.h     10 #include <llvm/DebugInfo/Symbolize/Symbolize.h>
 11 #include <llvm/Support/TargetSelect.h>             11 #include <llvm/Support/TargetSelect.h>
 12 #pragma GCC diagnostic pop                         12 #pragma GCC diagnostic pop
 13                                                    13 
 14 #include <inttypes.h>                              14 #include <inttypes.h>
 15 #include <stdio.h>                                 15 #include <stdio.h>
 16 #include <sys/types.h>                             16 #include <sys/types.h>
 17 #include <linux/compiler.h>                        17 #include <linux/compiler.h>
 18 extern "C" {                                       18 extern "C" {
 19 #include <linux/zalloc.h>                          19 #include <linux/zalloc.h>
 20 }                                                  20 }
 21 #include "symbol_conf.h"                           21 #include "symbol_conf.h"
 22 #include "llvm-c-helpers.h"                        22 #include "llvm-c-helpers.h"
 23                                                    23 
 24 extern "C"                                         24 extern "C"
 25 char *dso__demangle_sym(struct dso *dso, int k     25 char *dso__demangle_sym(struct dso *dso, int kmodule, const char *elf_name);
 26                                                    26 
 27 using namespace llvm;                              27 using namespace llvm;
 28 using llvm::symbolize::LLVMSymbolizer;             28 using llvm::symbolize::LLVMSymbolizer;
 29                                                    29 
 30 /*                                                 30 /*
 31  * Allocate a static LLVMSymbolizer, which wil     31  * Allocate a static LLVMSymbolizer, which will live to the end of the program.
 32  * Unlike the bfd paths, LLVMSymbolizer has it     32  * Unlike the bfd paths, LLVMSymbolizer has its own cache, so we do not need
 33  * to store anything in the dso struct.            33  * to store anything in the dso struct.
 34  */                                                34  */
 35 static LLVMSymbolizer *get_symbolizer()            35 static LLVMSymbolizer *get_symbolizer()
 36 {                                                  36 {
 37         static LLVMSymbolizer *instance = null     37         static LLVMSymbolizer *instance = nullptr;
 38         if (instance == nullptr) {                 38         if (instance == nullptr) {
 39                 LLVMSymbolizer::Options opts;      39                 LLVMSymbolizer::Options opts;
 40                 /*                                 40                 /*
 41                  * LLVM sometimes demangles sl     41                  * LLVM sometimes demangles slightly different from the rest
 42                  * of the code, and this misma     42                  * of the code, and this mismatch can cause new_inline_sym()
 43                  * to get confused and mark no     43                  * to get confused and mark non-inline symbol as inlined
 44                  * (since the name does not pr     44                  * (since the name does not properly match up with base_sym).
 45                  * Thus, disable the demanglin     45                  * Thus, disable the demangling and let the rest of the code
 46                  * handle it.                      46                  * handle it.
 47                  */                                47                  */
 48                 opts.Demangle = false;             48                 opts.Demangle = false;
 49                 instance = new LLVMSymbolizer(     49                 instance = new LLVMSymbolizer(opts);
 50         }                                          50         }
 51         return instance;                           51         return instance;
 52 }                                                  52 }
 53                                                    53 
 54 /* Returns 0 on error, 1 on success. */            54 /* Returns 0 on error, 1 on success. */
 55 static int extract_file_and_line(const DILineI     55 static int extract_file_and_line(const DILineInfo &line_info, char **file,
 56                                  unsigned int      56                                  unsigned int *line)
 57 {                                                  57 {
 58         if (file) {                                58         if (file) {
 59                 if (line_info.FileName == "<in     59                 if (line_info.FileName == "<invalid>") {
 60                         /* Match the conventio     60                         /* Match the convention of libbfd. */
 61                         *file = nullptr;           61                         *file = nullptr;
 62                 } else {                           62                 } else {
 63                         /* The caller expects      63                         /* The caller expects to get something it can free(). */
 64                         *file = strdup(line_in     64                         *file = strdup(line_info.FileName.c_str());
 65                         if (*file == nullptr)      65                         if (*file == nullptr)
 66                                 return 0;          66                                 return 0;
 67                 }                                  67                 }
 68         }                                          68         }
 69         if (line)                                  69         if (line)
 70                 *line = line_info.Line;            70                 *line = line_info.Line;
 71         return 1;                                  71         return 1;
 72 }                                                  72 }
 73                                                    73 
 74 extern "C"                                         74 extern "C"
 75 int llvm_addr2line(const char *dso_name, u64 a     75 int llvm_addr2line(const char *dso_name, u64 addr,
 76                    char **file, unsigned int *     76                    char **file, unsigned int *line,
 77                    bool unwind_inlines,            77                    bool unwind_inlines,
 78                    llvm_a2l_frame **inline_fra     78                    llvm_a2l_frame **inline_frames)
 79 {                                                  79 {
 80         LLVMSymbolizer *symbolizer = get_symbo     80         LLVMSymbolizer *symbolizer = get_symbolizer();
 81         object::SectionedAddress sectioned_add     81         object::SectionedAddress sectioned_addr = {
 82                 addr,                              82                 addr,
 83                 object::SectionedAddress::Unde     83                 object::SectionedAddress::UndefSection
 84         };                                         84         };
 85                                                    85 
 86         if (unwind_inlines) {                      86         if (unwind_inlines) {
 87                 Expected<DIInliningInfo> res_o     87                 Expected<DIInliningInfo> res_or_err =
 88                         symbolizer->symbolizeI     88                         symbolizer->symbolizeInlinedCode(dso_name,
 89                                                    89                                                          sectioned_addr);
 90                 if (!res_or_err)                   90                 if (!res_or_err)
 91                         return 0;                  91                         return 0;
 92                 unsigned num_frames = res_or_e     92                 unsigned num_frames = res_or_err->getNumberOfFrames();
 93                 if (num_frames == 0)               93                 if (num_frames == 0)
 94                         return 0;                  94                         return 0;
 95                                                    95 
 96                 if (extract_file_and_line(res_     96                 if (extract_file_and_line(res_or_err->getFrame(0),
 97                                           file     97                                           file, line) == 0)
 98                         return 0;                  98                         return 0;
 99                                                    99 
100                 *inline_frames = (llvm_a2l_fra    100                 *inline_frames = (llvm_a2l_frame *)calloc(
101                         num_frames, sizeof(**i    101                         num_frames, sizeof(**inline_frames));
102                 if (*inline_frames == nullptr)    102                 if (*inline_frames == nullptr)
103                         return 0;                 103                         return 0;
104                                                   104 
105                 for (unsigned i = 0; i < num_f    105                 for (unsigned i = 0; i < num_frames; ++i) {
106                         const DILineInfo &src     106                         const DILineInfo &src = res_or_err->getFrame(i);
107                                                   107 
108                         llvm_a2l_frame &dst =     108                         llvm_a2l_frame &dst = (*inline_frames)[i];
109                         if (src.FileName == "<    109                         if (src.FileName == "<invalid>")
110                                 /* Match the c    110                                 /* Match the convention of libbfd. */
111                                 dst.filename =    111                                 dst.filename = nullptr;
112                         else                      112                         else
113                                 dst.filename =    113                                 dst.filename = strdup(src.FileName.c_str());
114                         dst.funcname = strdup(    114                         dst.funcname = strdup(src.FunctionName.c_str());
115                         dst.line = src.Line;      115                         dst.line = src.Line;
116                                                   116 
117                         if (dst.filename == nu    117                         if (dst.filename == nullptr ||
118                             dst.funcname == nu    118                             dst.funcname == nullptr) {
119                                 for (unsigned     119                                 for (unsigned j = 0; j <= i; ++j) {
120                                         zfree(    120                                         zfree(&(*inline_frames)[j].filename);
121                                         zfree(    121                                         zfree(&(*inline_frames)[j].funcname);
122                                 }                 122                                 }
123                                 zfree(inline_f    123                                 zfree(inline_frames);
124                                 return 0;         124                                 return 0;
125                         }                         125                         }
126                 }                                 126                 }
127                                                   127 
128                 return num_frames;                128                 return num_frames;
129         } else {                                  129         } else {
130                 if (inline_frames)                130                 if (inline_frames)
131                         *inline_frames = nullp    131                         *inline_frames = nullptr;
132                                                   132 
133                 Expected<DILineInfo> res_or_er    133                 Expected<DILineInfo> res_or_err =
134                         symbolizer->symbolizeC    134                         symbolizer->symbolizeCode(dso_name, sectioned_addr);
135                 if (!res_or_err)                  135                 if (!res_or_err)
136                         return 0;                 136                         return 0;
137                 return extract_file_and_line(*    137                 return extract_file_and_line(*res_or_err, file, line);
138         }                                         138         }
139 }                                                 139 }
140                                                   140 
141 static char *                                     141 static char *
142 make_symbol_relative_string(struct dso *dso, c    142 make_symbol_relative_string(struct dso *dso, const char *sym_name,
143                             u64 addr, u64 base    143                             u64 addr, u64 base_addr)
144 {                                                 144 {
145         if (!strcmp(sym_name, "<invalid>"))       145         if (!strcmp(sym_name, "<invalid>"))
146                 return NULL;                      146                 return NULL;
147                                                   147 
148         char *demangled = dso__demangle_sym(ds    148         char *demangled = dso__demangle_sym(dso, 0, sym_name);
149         if (base_addr && base_addr != addr) {     149         if (base_addr && base_addr != addr) {
150                 char buf[256];                    150                 char buf[256];
151                 snprintf(buf, sizeof(buf), "%s    151                 snprintf(buf, sizeof(buf), "%s+0x%" PRIx64,
152                          demangled ? demangled    152                          demangled ? demangled : sym_name, addr - base_addr);
153                 free(demangled);                  153                 free(demangled);
154                 return strdup(buf);               154                 return strdup(buf);
155         } else {                                  155         } else {
156                 if (demangled)                    156                 if (demangled)
157                         return demangled;         157                         return demangled;
158                 else                              158                 else
159                         return strdup(sym_name    159                         return strdup(sym_name);
160         }                                         160         }
161 }                                                 161 }
162                                                   162 
163 extern "C"                                        163 extern "C"
164 char *llvm_name_for_code(struct dso *dso, cons    164 char *llvm_name_for_code(struct dso *dso, const char *dso_name, u64 addr)
165 {                                                 165 {
166         LLVMSymbolizer *symbolizer = get_symbo    166         LLVMSymbolizer *symbolizer = get_symbolizer();
167         object::SectionedAddress sectioned_add    167         object::SectionedAddress sectioned_addr = {
168                 addr,                             168                 addr,
169                 object::SectionedAddress::Unde    169                 object::SectionedAddress::UndefSection
170         };                                        170         };
171         Expected<DILineInfo> res_or_err =         171         Expected<DILineInfo> res_or_err =
172                 symbolizer->symbolizeCode(dso_    172                 symbolizer->symbolizeCode(dso_name, sectioned_addr);
173         if (!res_or_err) {                        173         if (!res_or_err) {
174                 return NULL;                      174                 return NULL;
175         }                                         175         }
176         return make_symbol_relative_string(       176         return make_symbol_relative_string(
177                 dso, res_or_err->FunctionName.    177                 dso, res_or_err->FunctionName.c_str(),
178                 addr, res_or_err->StartAddress    178                 addr, res_or_err->StartAddress ? *res_or_err->StartAddress : 0);
179 }                                                 179 }
180                                                   180 
181 extern "C"                                        181 extern "C"
182 char *llvm_name_for_data(struct dso *dso, cons    182 char *llvm_name_for_data(struct dso *dso, const char *dso_name, u64 addr)
183 {                                                 183 {
184         LLVMSymbolizer *symbolizer = get_symbo    184         LLVMSymbolizer *symbolizer = get_symbolizer();
185         object::SectionedAddress sectioned_add    185         object::SectionedAddress sectioned_addr = {
186                 addr,                             186                 addr,
187                 object::SectionedAddress::Unde    187                 object::SectionedAddress::UndefSection
188         };                                        188         };
189         Expected<DIGlobal> res_or_err =           189         Expected<DIGlobal> res_or_err =
190                 symbolizer->symbolizeData(dso_    190                 symbolizer->symbolizeData(dso_name, sectioned_addr);
191         if (!res_or_err) {                        191         if (!res_or_err) {
192                 return NULL;                      192                 return NULL;
193         }                                         193         }
194         return make_symbol_relative_string(       194         return make_symbol_relative_string(
195                 dso, res_or_err->Name.c_str(),    195                 dso, res_or_err->Name.c_str(),
196                 addr, res_or_err->Start);         196                 addr, res_or_err->Start);
197 }                                                 197 }
198                                                   198 

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

kernel.org | git.kernel.org | LWN.net | Project Home | SVN repository | Mail admin

Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.

sflogo.php