X-Git-Url: http://info.iut-bm.univ-fcomte.fr/pub/gitweb/simgrid.git/blobdiff_plain/a5c5765ba9e8336b55101d248a8f43fbf9cd0c0c..4a69abcc786d029bd2962537f767d12a0f808d11:/src/xbt/exception.cpp diff --git a/src/xbt/exception.cpp b/src/xbt/exception.cpp index 6e9cf0c258..ad829c2169 100644 --- a/src/xbt/exception.cpp +++ b/src/xbt/exception.cpp @@ -4,9 +4,15 @@ /* This program is free software; you can redistribute it and/or modify it * under the terms of the license (GNU LGPL) which comes with this package. */ +#include + +#include #include +#include #include +#include #include +#include #include #include @@ -21,7 +27,7 @@ XBT_LOG_NEW_DEFAULT_SUBCATEGORY(xbt_exception, xbt, "Exceptions"); namespace simgrid { namespace xbt { -WithContextException::~WithContextException() {} +WithContextException::~WithContextException() = default; void logException( e_xbt_log_priority_t prio, @@ -68,5 +74,61 @@ 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); + }); +} + } }