14#include "arcane/utils/PlatformUtils.h"
15#include "arcane/utils/IStackTraceService.h"
16#include "arcane/utils/StackTrace.h"
17#include "arcane/utils/StringBuilder.h"
18#include "arcane/utils/Array.h"
19#include "arcane/utils/Process.h"
21#include "arcane/utils/ISymbolizerService.h"
22#include "arcane/ServiceBuilder.h"
23#include "arcane/Directory.h"
25#include "arcane/ServiceFactory.h"
26#include "arcane/AbstractService.h"
54class LibUnwindStackTraceService
76 unw_word_t m_base_ip = 0;
91 m_want_gdb_info =
true;
93 m_use_backtrace =
true;
104 using ProcInfoMap = std::map<unw_word_t,ProcInfo>;
105 ProcInfoMap m_proc_name_map;
106 std::mutex m_proc_name_map_mutex;
108 bool m_want_gdb_info;
109 bool m_use_backtrace;
111 ProcInfo _getFuncInfo(unw_word_t ip,unw_cursor_t* cursor);
112 ProcInfo _getFuncInfo(
void* ip);
122LibUnwindStackTraceService::ProcInfo LibUnwindStackTraceService::
123_getFuncInfo(unw_word_t ip,unw_cursor_t* cursor)
126 std::lock_guard<std::mutex> lk(m_proc_name_map_mutex);
127 auto v = m_proc_name_map.find(ip);
128 if (v!=m_proc_name_map.end())
133 char func_name_buf[10000];
134 char demangled_func_name_buf[10000];
135 unw_get_proc_name(cursor,func_name_buf,10000,&offset);
138 char* buf = abi::__cxa_demangle (func_name_buf,demangled_func_name_buf,&len,&dstatus);
140 pi.m_base_ip = offset;
143 void* addr = (
void*)ip;
144 int r2 = dladdr(addr,&dl_info);
146 const char* dli_fname = dl_info.dli_fname;
148 void* dli_fbase = dl_info.dli_fbase;
149 pi.m_file_name = dli_fname;
150 pi.m_file_loaded_address = (unw_word_t)dli_fbase;
155 pi.m_name = std::string_view(buf);
157 pi.m_name = std::string_view(func_name_buf);
159 std::lock_guard<std::mutex> lk(m_proc_name_map_mutex);
160 m_proc_name_map.insert(ProcInfoMap::value_type(ip,pi));
172_getFuncInfo(
void* addr)
175 std::lock_guard<std::mutex> lk(m_proc_name_map_mutex);
176 auto v = m_proc_name_map.find((unw_word_t)addr);
177 if (v!=m_proc_name_map.end()){
181 const size_t buf_size = 10000;
182 char demangled_func_name_buf[buf_size];
184 int r = dladdr(addr,&dl_info);
187 std::cout <<
"ERROR in dladdr\n";
190 const char* dli_sname = dl_info.dli_sname;
194 size_t len = buf_size;
195 char* buf = abi::__cxa_demangle (dli_sname,demangled_func_name_buf,&len,&dstatus);
199 pi.
m_name = std::string_view(buf);
201 pi.
m_name = std::string_view(dli_sname);
203 std::lock_guard<std::mutex> lk(m_proc_name_map_mutex);
204 m_proc_name_map.insert(ProcInfoMap::value_type((unw_word_t)addr,pi));
212String LibUnwindStackTraceService::
219 fprintf (stderr,
"\nNative stacktrace:\n\n");
221 size = backtrace (array, 256);
222 names = backtrace_symbols (array, size);
223 for (i =0; i < size; ++i) {
224 fprintf (stderr,
"\t%s\n", names [i]);
243 last_str = _getGDBStack();
248 const size_t hexa_buf_size = 100;
249 char hexa[hexa_buf_size+1];
252 unw_init_local(&cursor, &uc);
253 int current_func = 0;
257 while (unw_step(&cursor) > 0) {
259 unw_get_reg(&cursor, UNW_REG_IP, &ip);
260 if (current_func>=first_function){
261 ProcInfo pi = _getFuncInfo(ip,&cursor);
264 snprintf(hexa,hexa_buf_size,
"%14llx",(
long long)ip);
267 message += func_name;
281 last_str = ss->
stackTrace(backtrace_stack_frames.view());
284 message +=
"\nFileAndOffsetStack:{{\n";
309 last_str = _getGDBStack();
312 unw_init_local(&cursor, &uc);
313 int current_func = 0;
316 while (unw_step(&cursor) > 0) {
317 unw_get_reg(&cursor, UNW_REG_IP, &ip);
318 if (current_func==function_index){
319 ProcInfo pi = _getFuncInfo(ip,&cursor);
321 message += func_name;
337 Integer size = backtrace (ips, 256);
340 for(
Integer i=first_function; i<size; ++i ){
353 const size_t buf_size = 100;
354 char hexa[buf_size+1];
359 intptr_t ip = f.address();
363 snprintf(hexa,buf_size,
"%10llx",(
long long)ip);
366 message += func_name;
387 const size_t buf_size = 100;
388 char hexa[buf_size+1];
393 intptr_t ip = f.address();
395 message += (
pinfo.fileName() ?
pinfo.fileName() :
"()");
397 intptr_t file_base_address =
pinfo.m_file_loaded_address;
402 if (file_base_address==0x400000)
403 file_base_address = 0;
404 intptr_t offset_ip = (ip - file_base_address);
405 snprintf(hexa,buf_size,
"%llx",(
long long)offset_ip);
419class LLVMSymbolizerService
426 virtual ~LLVMSymbolizerService() {}
435 String m_llvm_symbolizer_path;
436 bool m_is_check_done;
446 String fullpath = dir.
file(
"llvm-symbolizer");
448 m_llvm_symbolizer_path = fullpath;
451 m_is_check_done =
true;
464 std::stringstream ostr;
470 intptr_t addr = frames[i].address();
471 int r2 = dladdr((
void*)addr,&dl_info);
472 const char* dli_fname =
nullptr;
473 intptr_t base_address = 0;
475 dli_fname = dl_info.dli_fname;
477 void* dli_fbase = dl_info.dli_fbase;
478 intptr_t true_base = (intptr_t)dli_fbase;
483 if (true_base==0x400000)
485 base_address = addr - true_base;
488 ostr << (dli_fname ? dli_fname :
"??") <<
" " << base_address <<
'\n';
491 std::string input_str(ostr.str());
494 args.setCommand(m_llvm_symbolizer_path);
497 args.setInputBytes(input_bytes);
498 args.addArguments(
"--demangle");
499 args.addArguments(
"--pretty-print");
#define ARCANE_SERVICE_INTERFACE(ainterface)
Macro pour déclarer une interface lors de l'enregistrement d'un service.
AbstractService(const ServiceBuildInfo &)
Constructeur à partir d'un ServiceBuildInfo.
Vue constante d'un tableau de type T.
constexpr Integer size() const noexcept
Nombre d'éléments du tableau.
Classe gérant un répertoire.
virtual String file(const String &file_name) const
Retourne le chemin complet du fichier file_name dans le répertoire.
Conserve une liste de taille fixe maximale de StackFrame.
void addFrame(const StackFrame &frame)
Ajoute frame à la liste des frames. Si nbFrame() est supérieur ou égal à MAX_FRAME,...
Interface de l'application.
Interface d'un service de trace des appels de fonctions.
Interface d'un service de récupération des symboles du code source.
virtual String stackTrace(ConstArrayView< StackFrame > frames)=0
Informations pour la pile d'appel frames.
String stackTrace(ConstArrayView< StackFrame > frames) override
Informations pour la pile d'appel frames.
void _checkValid()
Vérifie que le chemin spécifié est valid.
void build() override
Construction de niveau build du service.
Service de trace des appels de fonctions utilisant la libunwind.
StackTrace _backtraceStackTrace(const FixedStackFrameArray &stack_frames)
Pile d'appel via la fonction backtrace.
FixedStackFrameArray _backtraceStackFrame(int first_function)
Pile d'appel via la fonction backtrace.
String _generateFileAndOffset(const FixedStackFrameArray &stack_frames)
Génère liste des noms de fichier et offset d'une pile d'appel.
void build() override
Construction de niveau build du service.
StackTrace stackTrace(int first_function) override
Chaîne de caractère indiquant la pile d'appel.
StackTrace stackTraceFunction(int function_index) override
Retourne la pile à d'appel actuelle.
ByteConstArrayView outputBytes() const
Contient le résultat de la sortie standard (STDOUT) du processus.
static ProcessExecArgs::ExecStatus execute(ProcessExecArgs &args)
Exécute un processus dont les infos sont contenues dans args.
IApplication * application() const
Accès à l'application IApplication associé.
Structure contenant les informations pour créer un service.
Propriétés de création d'un service.
Conserve les adresses correspondantes à une pile d'appel. Cette classe est interne et ne dois pas êtr...
Informations sur la pile d'appel des fonctions.
Constructeur de chaîne de caractère unicode.
String toString() const
Retourne la chaîne de caractères construite.
Chaîne de caractères unicode.
TraceMessage pinfo() const
Flot pour un message d'information en parallèle.
#define ARCANE_REGISTER_SERVICE(aclass, a_service_property,...)
Macro pour enregistrer un service.
Integer len(const char *s)
Retourne la longueur de la chaîne s.
-*- tab-width: 2; indent-tabs-mode: nil; coding: utf-8-with-signature -*-
Integer arcaneCheckArraySize(unsigned long long size)
Vérifie que size peut être converti dans un 'Integer' pour servir de taille à un tableau....
std::int64_t Int64
Type entier signé sur 64 bits.
Int32 Integer
Type représentant un entier.
@ ST_Application
Le service s'utilise au niveau de l'application.
ConstArrayView< Byte > ByteConstArrayView
Equivalent C d'un tableau à une dimension de caractères.
unsigned char Byte
Type d'un octet.
const char * m_file_name
Nom de la bibliothèque (.so ou .exe) dans laquelle se trouve la méthode.
const char * fileName() const
Nom du fichier contenant la procédure. Peut-être nul.
unw_word_t m_file_loaded_address
Adresse de chargement de la bibliothèque.
String m_name
Nom (démanglé) de la procédure.