X-Git-Url: http://bilbo.iut-bm.univ-fcomte.fr/pub/gitweb/simgrid.git/blobdiff_plain/a5c5765ba9e8336b55101d248a8f43fbf9cd0c0c..09ae311009060c8bf6cda09873b78f81b268badd:/src/xbt/exception.cpp diff --git a/src/xbt/exception.cpp b/src/xbt/exception.cpp index 6e9cf0c258..49f6188893 100644 --- a/src/xbt/exception.cpp +++ b/src/xbt/exception.cpp @@ -7,6 +7,7 @@ #include #include #include +#include #include #include @@ -68,5 +69,66 @@ void logException( } } +static void showBacktrace(std::vector& bt) +{ + std::vector res = resolveBacktrace(&bt[0], bt.size()); + XBT_LOG(xbt_log_priority_critical, "Current backtrace:"); + for (std::string const& s : res) + XBT_LOG(xbt_log_priority_critical, " -> %s", s.c_str()); +} + +static std::terminate_handler previous_terminate_handler = nullptr; + +static void handler() +{ + // Avoid doing crazy things if we get an uncaught exception inside + // an uncaught exception + static std::atomic_flag lock = ATOMIC_FLAG_INIT; + if (lock.test_and_set()) { + XBT_ERROR("Multiple uncaught exceptions"); + std::abort(); + } + + // Get the current backtrace and exception + auto e = std::current_exception(); + auto bt = backtrace(); + try { + std::rethrow_exception(e); + } + + // We manage C++ exception ourselves + catch (std::exception& e) { + logException(xbt_log_priority_critical, "Uncaught exception", e); + showBacktrace(bt); + std::abort(); + } + + // We don't know how to manage other exceptions + catch (...) { + // If there was another handler let's delegate to it + if (previous_terminate_handler) + previous_terminate_handler(); + else { + XBT_ERROR("Unknown uncaught exception"); + showBacktrace(bt); + std::abort(); + } + } + +} + +void installExceptionHandler() +{ + static std::once_flag handler_flag; + std::call_once(handler_flag, [] { + previous_terminate_handler = std::set_terminate(handler); + }); +} + } } + +void xbt_set_terminate() +{ + simgrid::xbt::installExceptionHandler(); +}