X-Git-Url: http://info.iut-bm.univ-fcomte.fr/pub/gitweb/simgrid.git/blobdiff_plain/931c72e9293d64a1797a6187dd0e744408c23d59..1873a02acdc52506c010f43b4c78a8b8400dc0de:/src/xbt/exception.cpp diff --git a/src/xbt/exception.cpp b/src/xbt/exception.cpp index 6e9cf0c258..a1c6d0b52b 100644 --- a/src/xbt/exception.cpp +++ b/src/xbt/exception.cpp @@ -1,14 +1,22 @@ -/* Copyright (c) 2005-2016. The SimGrid Team. +/* Copyright (c) 2005-2017. The SimGrid Team. * All rights reserved. */ /* 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 +#include #include #include #include @@ -21,7 +29,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, @@ -41,7 +49,7 @@ void logException( XBT_LOG(prio, "%s %s: %s", context, name.get(), exception.what()); // Do we have a backtrace? - if (with_context != nullptr) { + if (with_context != nullptr && not xbt_cfg_get_boolean("exception/cutpath")) { auto backtrace = simgrid::xbt::resolveBacktrace( with_context->backtrace().data(), with_context->backtrace().size()); for (std::string const& s : backtrace) @@ -68,5 +76,65 @@ void logException( } } +static void showBacktrace(std::vector& bt) +{ + if (xbt_cfg_get_boolean("exception/cutpath")) { + XBT_LOG(xbt_log_priority_critical, "Display of current backtrace disabled by --cfg=exception/cutpath."); + return; + } + 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("Handling an exception raised an exception. Bailing out."); + 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); + }); +} + } }