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 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 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 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 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 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 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 1420eae32dcSDimitry Andric void __sanitizer_symbolize_flush() { 1430eae32dcSDimitry Andric if (Symbolizer) 1440eae32dcSDimitry Andric Symbolizer->flush(); 1450eae32dcSDimitry Andric } 14668d75effSDimitry Andric 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 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 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. 195fe6060f1SDimitry Andric int __cxa_atexit(void (*f)(void *a), void *arg, void *dso) { return 0; } 196fe6060f1SDimitry Andric 19768d75effSDimitry Andric } // extern "C" 198