168d75effSDimitry Andric //===-- sanitizer_symbolize.cpp ---------------------------------*- C++ -*-===//
268d75effSDimitry Andric //
368d75effSDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
468d75effSDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
568d75effSDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
668d75effSDimitry Andric //
768d75effSDimitry Andric //===----------------------------------------------------------------------===//
868d75effSDimitry Andric //
968d75effSDimitry Andric // Implementation of weak hooks from sanitizer_symbolizer_posix_libcdep.cpp.
1068d75effSDimitry Andric //
1168d75effSDimitry Andric //===----------------------------------------------------------------------===//
1268d75effSDimitry Andric
1381ad6265SDimitry Andric #include <inttypes.h>
1468d75effSDimitry Andric #include <stdio.h>
155ffd83dbSDimitry Andric
1668d75effSDimitry Andric #include <string>
1768d75effSDimitry Andric
1868d75effSDimitry Andric #include "llvm/DebugInfo/Symbolize/DIPrinter.h"
1968d75effSDimitry Andric #include "llvm/DebugInfo/Symbolize/Symbolize.h"
205f757f3fSDimitry Andric #include "llvm/Demangle/Demangle.h"
2168d75effSDimitry Andric
220eae32dcSDimitry Andric static llvm::symbolize::LLVMSymbolizer *Symbolizer = nullptr;
230eae32dcSDimitry Andric static bool Demangle = true;
240eae32dcSDimitry Andric static bool InlineFrames = true;
250eae32dcSDimitry Andric
getDefaultSymbolizer()2668d75effSDimitry Andric static llvm::symbolize::LLVMSymbolizer *getDefaultSymbolizer() {
270eae32dcSDimitry Andric if (Symbolizer)
280eae32dcSDimitry Andric return Symbolizer;
290eae32dcSDimitry Andric llvm::symbolize::LLVMSymbolizer::Options Opts;
300eae32dcSDimitry Andric Opts.Demangle = Demangle;
315f757f3fSDimitry Andric Opts.UntagAddresses = true;
320eae32dcSDimitry Andric Symbolizer = new llvm::symbolize::LLVMSymbolizer(Opts);
330eae32dcSDimitry Andric return Symbolizer;
3468d75effSDimitry Andric }
3568d75effSDimitry Andric
getDefaultPrinterConfig()36fe6060f1SDimitry Andric static llvm::symbolize::PrinterConfig getDefaultPrinterConfig() {
37fe6060f1SDimitry Andric llvm::symbolize::PrinterConfig Config;
38fe6060f1SDimitry Andric Config.Pretty = false;
39fe6060f1SDimitry Andric Config.Verbose = false;
40fe6060f1SDimitry Andric Config.PrintFunctions = true;
41fe6060f1SDimitry Andric Config.PrintAddress = false;
42fe6060f1SDimitry Andric Config.SourceContextLines = 0;
43fe6060f1SDimitry Andric return Config;
44fe6060f1SDimitry Andric }
45fe6060f1SDimitry Andric
symbolize_error_handler(llvm::raw_string_ostream & OS)4606c3fb27SDimitry Andric static llvm::symbolize::ErrorHandler symbolize_error_handler(
4706c3fb27SDimitry Andric llvm::raw_string_ostream &OS) {
4806c3fb27SDimitry Andric return
4906c3fb27SDimitry Andric [&](const llvm::ErrorInfoBase &ErrorInfo, llvm::StringRef ErrorBanner) {
5006c3fb27SDimitry Andric OS << ErrorBanner;
5106c3fb27SDimitry Andric ErrorInfo.log(OS);
5206c3fb27SDimitry Andric OS << '\n';
5306c3fb27SDimitry Andric };
5406c3fb27SDimitry Andric }
5506c3fb27SDimitry Andric
5668d75effSDimitry Andric namespace __sanitizer {
575f757f3fSDimitry Andric int internal_snprintf(char *buffer, uintptr_t length, const char *format, ...);
5868d75effSDimitry Andric } // namespace __sanitizer
5968d75effSDimitry Andric
6068d75effSDimitry Andric extern "C" {
6168d75effSDimitry Andric
6268d75effSDimitry Andric typedef uint64_t u64;
6368d75effSDimitry Andric
__sanitizer_symbolize_code(const char * ModuleName,uint64_t ModuleOffset,char * Buffer,int MaxLength)6468d75effSDimitry Andric bool __sanitizer_symbolize_code(const char *ModuleName, uint64_t ModuleOffset,
650eae32dcSDimitry Andric char *Buffer, int MaxLength) {
6668d75effSDimitry Andric std::string Result;
6768d75effSDimitry Andric {
6868d75effSDimitry Andric llvm::raw_string_ostream OS(Result);
69fe6060f1SDimitry Andric llvm::symbolize::PrinterConfig Config = getDefaultPrinterConfig();
70fe6060f1SDimitry Andric llvm::symbolize::Request Request{ModuleName, ModuleOffset};
7106c3fb27SDimitry Andric auto Printer = std::make_unique<llvm::symbolize::LLVMPrinter>(
7206c3fb27SDimitry Andric OS, symbolize_error_handler(OS), Config);
73fe6060f1SDimitry Andric
74*cb14a3feSDimitry Andric // TODO: it is necessary to set proper SectionIndex here.
7568d75effSDimitry Andric // object::SectionedAddress::UndefSection works for only absolute addresses.
760eae32dcSDimitry Andric if (InlineFrames) {
7768d75effSDimitry Andric auto ResOrErr = getDefaultSymbolizer()->symbolizeInlinedCode(
7868d75effSDimitry Andric ModuleName,
7968d75effSDimitry Andric {ModuleOffset, llvm::object::SectionedAddress::UndefSection});
805f757f3fSDimitry Andric if (!ResOrErr)
815f757f3fSDimitry Andric return false;
825f757f3fSDimitry Andric Printer->print(Request, ResOrErr.get());
835ffd83dbSDimitry Andric } else {
845ffd83dbSDimitry Andric auto ResOrErr = getDefaultSymbolizer()->symbolizeCode(
855ffd83dbSDimitry Andric ModuleName,
865ffd83dbSDimitry Andric {ModuleOffset, llvm::object::SectionedAddress::UndefSection});
875f757f3fSDimitry Andric if (!ResOrErr)
885f757f3fSDimitry Andric return false;
895f757f3fSDimitry Andric Printer->print(Request, ResOrErr.get());
905ffd83dbSDimitry Andric }
9168d75effSDimitry Andric }
9268d75effSDimitry Andric return __sanitizer::internal_snprintf(Buffer, MaxLength, "%s",
9368d75effSDimitry Andric Result.c_str()) < MaxLength;
9468d75effSDimitry Andric }
9568d75effSDimitry Andric
__sanitizer_symbolize_data(const char * ModuleName,uint64_t ModuleOffset,char * Buffer,int MaxLength)9668d75effSDimitry Andric bool __sanitizer_symbolize_data(const char *ModuleName, uint64_t ModuleOffset,
9768d75effSDimitry Andric char *Buffer, int MaxLength) {
9868d75effSDimitry Andric std::string Result;
9968d75effSDimitry Andric {
100fe6060f1SDimitry Andric llvm::symbolize::PrinterConfig Config = getDefaultPrinterConfig();
10168d75effSDimitry Andric llvm::raw_string_ostream OS(Result);
102fe6060f1SDimitry Andric llvm::symbolize::Request Request{ModuleName, ModuleOffset};
10306c3fb27SDimitry Andric auto Printer = std::make_unique<llvm::symbolize::LLVMPrinter>(
10406c3fb27SDimitry Andric OS, symbolize_error_handler(OS), Config);
105fe6060f1SDimitry Andric
106*cb14a3feSDimitry Andric // TODO: it is necessary to set proper SectionIndex here.
10768d75effSDimitry Andric // object::SectionedAddress::UndefSection works for only absolute addresses.
10868d75effSDimitry Andric auto ResOrErr = getDefaultSymbolizer()->symbolizeData(
10968d75effSDimitry Andric ModuleName,
11068d75effSDimitry Andric {ModuleOffset, llvm::object::SectionedAddress::UndefSection});
1115f757f3fSDimitry Andric if (!ResOrErr)
1125f757f3fSDimitry Andric return false;
1135f757f3fSDimitry Andric Printer->print(Request, ResOrErr.get());
1145f757f3fSDimitry Andric }
1155f757f3fSDimitry Andric return __sanitizer::internal_snprintf(Buffer, MaxLength, "%s",
1165f757f3fSDimitry Andric Result.c_str()) < MaxLength;
1175f757f3fSDimitry Andric }
1185f757f3fSDimitry Andric
__sanitizer_symbolize_frame(const char * ModuleName,uint64_t ModuleOffset,char * Buffer,int MaxLength)1195f757f3fSDimitry Andric bool __sanitizer_symbolize_frame(const char *ModuleName, uint64_t ModuleOffset,
1205f757f3fSDimitry Andric char *Buffer, int MaxLength) {
1215f757f3fSDimitry Andric std::string Result;
1225f757f3fSDimitry Andric {
1235f757f3fSDimitry Andric llvm::symbolize::PrinterConfig Config = getDefaultPrinterConfig();
1245f757f3fSDimitry Andric llvm::raw_string_ostream OS(Result);
1255f757f3fSDimitry Andric llvm::symbolize::Request Request{ModuleName, ModuleOffset};
1265f757f3fSDimitry Andric auto Printer = std::make_unique<llvm::symbolize::LLVMPrinter>(
1275f757f3fSDimitry Andric OS, symbolize_error_handler(OS), Config);
1285f757f3fSDimitry Andric
129*cb14a3feSDimitry Andric // TODO: it is necessary to set proper SectionIndex here.
1305f757f3fSDimitry Andric // object::SectionedAddress::UndefSection works for only absolute addresses.
1315f757f3fSDimitry Andric auto ResOrErr = getDefaultSymbolizer()->symbolizeFrame(
1325f757f3fSDimitry Andric ModuleName,
1335f757f3fSDimitry Andric {ModuleOffset, llvm::object::SectionedAddress::UndefSection});
1345f757f3fSDimitry Andric if (!ResOrErr)
1355f757f3fSDimitry Andric return false;
1365f757f3fSDimitry Andric Printer->print(Request, ResOrErr.get());
13768d75effSDimitry Andric }
13868d75effSDimitry Andric return __sanitizer::internal_snprintf(Buffer, MaxLength, "%s",
13968d75effSDimitry Andric Result.c_str()) < MaxLength;
14068d75effSDimitry Andric }
14168d75effSDimitry Andric
__sanitizer_symbolize_flush()1420eae32dcSDimitry Andric void __sanitizer_symbolize_flush() {
1430eae32dcSDimitry Andric if (Symbolizer)
1440eae32dcSDimitry Andric Symbolizer->flush();
1450eae32dcSDimitry Andric }
14668d75effSDimitry Andric
__sanitizer_symbolize_demangle(const char * Name,char * Buffer,int MaxLength)1475f757f3fSDimitry Andric bool __sanitizer_symbolize_demangle(const char *Name, char *Buffer,
14868d75effSDimitry Andric int MaxLength) {
1495f757f3fSDimitry Andric std::string Result;
1505f757f3fSDimitry Andric if (!llvm::nonMicrosoftDemangle(Name, Result))
1515f757f3fSDimitry Andric return false;
15268d75effSDimitry Andric return __sanitizer::internal_snprintf(Buffer, MaxLength, "%s",
1535f757f3fSDimitry Andric Result.c_str()) < MaxLength;
15468d75effSDimitry Andric }
15568d75effSDimitry Andric
__sanitizer_symbolize_set_demangle(bool Value)1560eae32dcSDimitry Andric bool __sanitizer_symbolize_set_demangle(bool Value) {
1570eae32dcSDimitry Andric // Must be called before LLVMSymbolizer created.
1580eae32dcSDimitry Andric if (Symbolizer)
1590eae32dcSDimitry Andric return false;
1600eae32dcSDimitry Andric Demangle = Value;
1610eae32dcSDimitry Andric return true;
1620eae32dcSDimitry Andric }
1630eae32dcSDimitry Andric
__sanitizer_symbolize_set_inline_frames(bool Value)1640eae32dcSDimitry Andric bool __sanitizer_symbolize_set_inline_frames(bool Value) {
1650eae32dcSDimitry Andric InlineFrames = Value;
1660eae32dcSDimitry Andric return true;
1670eae32dcSDimitry Andric }
1680eae32dcSDimitry Andric
169fe6060f1SDimitry Andric // Override __cxa_atexit and ignore callbacks.
170fe6060f1SDimitry Andric // This prevents crashes in a configuration when the symbolizer
171fe6060f1SDimitry Andric // is built into sanitizer runtime and consequently into the test process.
172fe6060f1SDimitry Andric // LLVM libraries have some global objects destroyed during exit,
173fe6060f1SDimitry Andric // so if the test process triggers any bugs after that, the symbolizer crashes.
174fe6060f1SDimitry Andric // An example stack trace of such crash:
175fe6060f1SDimitry Andric //
176fe6060f1SDimitry Andric // #1 __cxa_throw
177fe6060f1SDimitry Andric // #2 std::__u::__throw_system_error
178fe6060f1SDimitry Andric // #3 std::__u::recursive_mutex::lock
179fe6060f1SDimitry Andric // #4 __sanitizer_llvm::ManagedStaticBase::RegisterManagedStatic
180fe6060f1SDimitry Andric // #5 __sanitizer_llvm::errorToErrorCode
181fe6060f1SDimitry Andric // #6 __sanitizer_llvm::getFileAux
182fe6060f1SDimitry Andric // #7 __sanitizer_llvm::MemoryBuffer::getFileOrSTDIN
183fe6060f1SDimitry Andric // #10 __sanitizer_llvm::symbolize::LLVMSymbolizer::getOrCreateModuleInfo
184fe6060f1SDimitry Andric // #13 __sanitizer::Symbolizer::SymbolizeData
185fe6060f1SDimitry Andric // #14 __tsan::SymbolizeData
186fe6060f1SDimitry Andric // #16 __tsan::ReportRace
187fe6060f1SDimitry Andric // #18 __tsan_write4
188fe6060f1SDimitry Andric // #19 race() () at test/tsan/atexit4.cpp
189fe6060f1SDimitry Andric // #20 cxa_at_exit_wrapper
190fe6060f1SDimitry Andric // #21 __cxa_finalize
191fe6060f1SDimitry Andric // #22 __do_fini
192fe6060f1SDimitry Andric //
193fe6060f1SDimitry Andric // For the standalone llvm-symbolizer this does not hurt,
194fe6060f1SDimitry Andric // we just don't destroy few global objects on exit.
__cxa_atexit(void (* f)(void * a),void * arg,void * dso)195fe6060f1SDimitry Andric int __cxa_atexit(void (*f)(void *a), void *arg, void *dso) { return 0; }
196fe6060f1SDimitry Andric
19768d75effSDimitry Andric } // extern "C"
198