Skip to content
Snippets Groups Projects
Commit ba526ce5 authored by Sebastian Hahta's avatar Sebastian Hahta
Browse files

Do not decode trace when it is not used.

parent 1a3995aa
No related branches found
No related tags found
1 merge request!309Decode trace symbols only when trace requested (GCC)
Pipeline #27103 passed
...@@ -7,29 +7,29 @@ namespace ftl { ...@@ -7,29 +7,29 @@ namespace ftl {
class Formatter { class Formatter {
public: public:
Formatter() {} Formatter() {}
~Formatter() {} ~Formatter() {}
template <typename Type> template <typename Type>
inline Formatter & operator << (const Type & value) inline Formatter & operator << (const Type & value)
{ {
stream_ << value; stream_ << value;
return *this; return *this;
} }
inline std::string str() const { return stream_.str(); } inline std::string str() const { return stream_.str(); }
inline operator std::string () const { return stream_.str(); } inline operator std::string () const { return stream_.str(); }
enum ConvertToString enum ConvertToString
{ {
to_str to_str
}; };
inline std::string operator >> (ConvertToString) { return stream_.str(); } inline std::string operator >> (ConvertToString) { return stream_.str(); }
private: private:
std::stringstream stream_; std::stringstream stream_;
Formatter(const Formatter &); Formatter(const Formatter &);
Formatter & operator = (Formatter &); Formatter & operator = (Formatter &);
}; };
class exception : public std::exception class exception : public std::exception
...@@ -39,22 +39,30 @@ class exception : public std::exception ...@@ -39,22 +39,30 @@ class exception : public std::exception
explicit exception(const Formatter &msg); explicit exception(const Formatter &msg);
~exception(); ~exception();
const char * what () const throw () { const char* what() const throw () {
processed_ = true; processed_ = true;
return msg_.c_str(); return msg_.c_str();
} }
const char * trace () const throw () { std::string trace() const throw () {
return trace_.c_str(); return decode_backtrace();
} }
void ignore() const { processed_ = true; } void ignore() const { processed_ = true; }
private: private:
std::string decode_backtrace() const;
std::string msg_; std::string msg_;
std::string trace_;
mutable bool processed_; mutable bool processed_;
#ifdef __GNUC__
static const int TRACE_SIZE_MAX_ = 16;
void* trace_[TRACE_SIZE_MAX_];
int trace_size_;
#endif
}; };
} }
#define FTL_Error(A) (ftl::exception(ftl::Formatter() << __FILE__ << ":" << __LINE__ << ": " << A)) #define FTL_Error(A) (ftl::exception(ftl::Formatter() << __FILE__ << ":" << __LINE__ << ": " << A))
......
...@@ -7,53 +7,52 @@ ...@@ -7,53 +7,52 @@
#include <execinfo.h> #include <execinfo.h>
#include <dlfcn.h> #include <dlfcn.h>
#include <cxxabi.h> #include <cxxabi.h>
#endif
using ftl::exception;
using std::string;
static std::string demangle(const char* name) { std::string demangle(const char* name) {
if (!name) { if (!name) {
return "[unknown symbol]"; return "[unknown symbol]";
} }
#ifdef __GNUC__
int status; int status;
char* demangled = abi::__cxa_demangle(name, NULL, 0, &status); char* demangled = abi::__cxa_demangle(name, NULL, 0, &status);
if (!demangled) { if (!demangled) {
return std::string(name); return std::string(name);
} }
else { else {
auto result = string(demangled); auto result = std::string(demangled);
free(demangled); free(demangled);
return result; return result;
} }
#else
return std::string(name);
#endif
} }
static std::string addr_to_string(const void* addr) { #endif
using ftl::exception;
using std::string;
string addr_to_string(const void* addr) {
std::stringstream ss; std::stringstream ss;
ss << addr; ss << addr;
return ss.str(); return ss.str();
} }
static std::string getBackTrace() {
#ifdef __GNUC__ #ifdef __GNUC__
string exception::decode_backtrace() const {
string result; string result;
void *trace[16]; // backtrace_symbols() as fallback (no data from dladdr())
int trace_size = 0; char **messages = backtrace_symbols(trace_, trace_size_);
trace_size = backtrace(trace, 16);
char **messages = backtrace_symbols(trace, trace_size);
result = "[bt] Trace:\n"; if (!messages) {
return string("[bt] no trace");
}
/* skip first stack frame (points here) */ /* skip first stack frame (points here) */
for (int i=2; i < trace_size; ++i) { for (int i=1; i < trace_size_; ++i) {
result += string("[bt] #") + std::to_string(i-1) + string(" "); result += string("[bt] #") + std::to_string(i-1)
+ string(TRACE_SIZE_MAX_/10 - (i-1)/10, ' ')
+ string(" ");
Dl_info info; Dl_info info;
if (dladdr(trace[i], &info) && info.dli_saddr) { if (dladdr(trace_[i], &info) && info.dli_saddr) {
auto name = demangle(info.dli_sname); auto name = demangle(info.dli_sname);
string fname = info.dli_fname ? info.dli_fname: "[unknown file]"; string fname = info.dli_fname ? info.dli_fname: "[unknown file]";
...@@ -71,25 +70,31 @@ static std::string getBackTrace() { ...@@ -71,25 +70,31 @@ static std::string getBackTrace() {
free(messages); free(messages);
return result; return result;
}
#elif _MSC_VER
return "";
#else #else
return ""; string exception::decode_backtrace() const {
#endif return string();
} }
#endif
exception::exception(const char *msg) : msg_(msg), processed_(false) { exception::exception(const char *msg) : msg_(msg), processed_(false) {
trace_ = std::move(getBackTrace()); #ifdef __GNUC__
trace_size_ = backtrace(trace_, TRACE_SIZE_MAX_);
#endif
} }
exception::exception(const ftl::Formatter &msg) : msg_(msg.str()), processed_(false) { exception::exception(const ftl::Formatter &msg) : msg_(msg.str()), processed_(false) {
trace_ = std::move(getBackTrace()); #ifdef __GNUC__
trace_size_ = backtrace(trace_, TRACE_SIZE_MAX_);
#endif
} }
exception::~exception() { exception::~exception() {
if (!processed_) { if (!processed_) {
LOG(ERROR) << "Unhandled exception: " << what(); LOG(ERROR) << "Unhandled exception: " << what();
LOG(ERROR) << trace_; #ifdef __GNUC__
LOG(ERROR) << "Trace:\n" << decode_backtrace();
#endif
} }
} }
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment