X-Git-Url: http://info.iut-bm.univ-fcomte.fr/pub/gitweb/simgrid.git/blobdiff_plain/fae10815035e4d0d150ff5fdbb7aa8fe25448c57..53300623ffc4a530dedcc517d559926ea0891edf:/include/xbt/exception.hpp?ds=sidebyside diff --git a/include/xbt/exception.hpp b/include/xbt/exception.hpp index e0807e3353..8fa032d966 100644 --- a/include/xbt/exception.hpp +++ b/include/xbt/exception.hpp @@ -1,4 +1,4 @@ -/* Copyright (c) 2005-2016. The SimGrid Team.All rights reserved. */ +/* Copyright (c) 2005-2018. 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. */ @@ -17,38 +17,65 @@ #include // xbt_procname #include // xbt_getpid +/** @addtogroup XBT_ex + * @brief Exceptions support + */ + namespace simgrid { namespace xbt { +/** A backtrace + * + * This is used (among other things) in exceptions to store the associated + * backtrace. + * + * @ingroup XBT_ex + */ typedef std::vector Backtrace; -struct ThrowPoint { - ThrowPoint() {} - ThrowPoint(const char* file, int line, const char* function) : - file(file), line(line), function(function) {} +/** The location of where an exception has been throwed + * + * This is a tuple (__FILE__, __LINE__, __func__) and can be created with + * @ref XBT_THROW_POINT. + * + * @ingroup XBT_ex + */ +class ThrowPoint { + public: + ThrowPoint() = default; + explicit ThrowPoint(const char* file, int line, const char* function) : file(file), line(line), function(function) {} const char* file = nullptr; int line = 0; const char* function = nullptr; }; -/** A polymorphic mixin class for adding context to an exception */ -XBT_PUBLIC_CLASS WithContextException { +/** Create a ThrowPoint with (__FILE__, __LINE__, __func__) */ +#define XBT_THROW_POINT ::simgrid::xbt::ThrowPoint(__FILE__, __LINE__, __func__) + +/** A base class for exceptions with context + * + * This is a base class for exceptions which store additional contextual + * infomations about them: backtrace, throw point, simulated process name + * and PID, etc. + * + * You are not expected to inherit from it. Instead of you use should + * @ref XBT_THROW an exception which will throw a subclass of your original + * exception with those additional features. + * + * However, you can try `dynamic_cast` an exception to this type in order to + * get contextual information about the exception. + */ +class XBT_PUBLIC WithContextException { public: WithContextException() : backtrace_(simgrid::xbt::backtrace()), procname_(xbt_procname()), pid_(xbt_getpid()) {} - WithContextException(Backtrace bt) : - backtrace_(std::move(bt)), - procname_(xbt_procname()), - pid_(xbt_getpid()) + explicit WithContextException(Backtrace bt) : backtrace_(std::move(bt)), procname_(xbt_procname()), pid_(xbt_getpid()) {} - WithContextException(ThrowPoint throwpoint, Backtrace bt) : - backtrace_(std::move(bt)), - procname_(xbt_procname()), - pid_(xbt_getpid()), - throwpoint_(throwpoint) + explicit WithContextException(ThrowPoint throwpoint, Backtrace bt) + : backtrace_(std::move(bt)), procname_(xbt_procname()), pid_(xbt_getpid()), throwpoint_(throwpoint) {} virtual ~WithContextException(); Backtrace const& backtrace() const @@ -65,13 +92,14 @@ private: ThrowPoint throwpoint_; }; -/** Internal class used to mixin the two classes */ +/** Internal class used to mixin an exception E with WithContextException */ template class WithContext : public E, public WithContextException { public: - WithContext(E exception) : - E(std::move(exception)) {} + static_assert(not std::is_base_of::value, "Trying to appli WithContext twice"); + + explicit WithContext(E exception) : E(std::move(exception)) {} WithContext(E exception, ThrowPoint throwpoint, Backtrace backtrace) : E(std::move(exception)), WithContextException(throwpoint, std::move(backtrace)) {} @@ -81,40 +109,24 @@ public: WithContext(E exception, WithContextException context) : E(std::move(exception)), WithContextException(std::move(context)) {} - ~WithContext() override {} + ~WithContext() override = default; }; -/** Throw a given exception a context +/** Throw a C++ exception with some context * - * @param exception exception to throw - * @param backtrace backtrace to attach + * @param e Exception to throw + * @ingroup XBT_ex */ -template -[[noreturn]] inline -typename std::enable_if< !std::is_base_of::value >::type -throwWithContext( - E exception, - // Thanks to the default argument, we are taking the backtrace in the caller: - Backtrace backtrace = simgrid::xbt::backtrace()) -{ - throw WithContext(std::move(exception), std::move(backtrace)); -} - -template -[[noreturn]] inline -typename std::enable_if< !std::is_base_of::value >::type -throwWithContext( - E exception, - ThrowPoint throwpoint, - // Thanks to the default argument, we are taking the backtrace in the caller: - Backtrace backtrace = simgrid::xbt::backtrace()) -{ - throw WithContext(std::move(exception), throwpoint, std::move(backtrace)); -} - -#define XBT_THROW_POINT ::simgrid::xbt::ThrowPoint(__FILE__, __LINE__, __func__) #define XBT_THROW(e) \ - ::simgrid::xbt::throwWithContext(std::move(e), XBT_THROW_POINT) + throw WithContext(std::move(exception), throwpoint, simgrid::xbt::backtrace()) + +/** Throw a C++ exception with a context and a nexted exception/cause + * + * @param e Exception to throw + * @ingroup XBT_ex + */ +#define XBT_THROW_NESTED(e) \ + std::throw_with_nested(WithContext(std::move(exception), throwpoint, simgrid::xbt::backtrace())) } }