LCOV - code coverage report
Current view: top level - tools - Exception.h (source / functions) Hit Total Coverage
Test: plumed test coverage Lines: 34 37 91.9 %
Date: 2025-03-25 09:33:27 Functions: 34 139 24.5 %

          Line data    Source code
       1             : /* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
       2             :    Copyright (c) 2012-2023 The plumed team
       3             :    (see the PEOPLE file at the root of the distribution for a list of names)
       4             : 
       5             :    See http://www.plumed.org for more information.
       6             : 
       7             :    This file is part of plumed, version 2.
       8             : 
       9             :    plumed is free software: you can redistribute it and/or modify
      10             :    it under the terms of the GNU Lesser General Public License as published by
      11             :    the Free Software Foundation, either version 3 of the License, or
      12             :    (at your option) any later version.
      13             : 
      14             :    plumed is distributed in the hope that it will be useful,
      15             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      16             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      17             :    GNU Lesser General Public License for more details.
      18             : 
      19             :    You should have received a copy of the GNU Lesser General Public License
      20             :    along with plumed.  If not, see <http://www.gnu.org/licenses/>.
      21             : +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
      22             : #ifndef __PLUMED_tools_Exception_h
      23             : #define __PLUMED_tools_Exception_h
      24             : 
      25             : #include <exception>
      26             : #include <string>
      27             : #include <stdexcept>
      28             : #include <sstream>
      29             : #include <array>
      30             : 
      31             : namespace PLMD {
      32             : 
      33             : /**
      34             : \ingroup TOOLBOX
      35             : Class to deal with Plumed runtime errors.
      36             : 
      37             : This class and the related macros can be used to detect programming
      38             : errors. Typical cases are internal inconsistencies or errors in the plumed<->MD
      39             : interface. Mistakes made by final users (i.e. in the `plumed.dat` file)
      40             : should probably be documented in some better way (e.g. printing parts of the manual in the output).
      41             : However, also this class allows for significant information to be attached.
      42             : Let's try to make error messages as informative as possible!
      43             : 
      44             : \note This class has been rewritten in PLUMED 2.5. It works in a backward compatible manner,
      45             : but is much more flexible. The main novelty is that we can use insertion operators to
      46             : add arbitrary messages, as in `plumed_error()<<"check this vector "<<v;`
      47             : See below for more details.
      48             : 
      49             : To throw an error, just throw a c++ exception
      50             : \verbatim
      51             :   if(something_bad) throw Exception();
      52             : \endverbatim
      53             : or better add an error message to that
      54             : \verbatim
      55             :   if(something_bad) throw Exception("describe the error here");
      56             : \endverbatim
      57             : 
      58             : As of PLUMED 2.5 you can add multiple messages, they will just be concatenated,
      59             : but to do se you should use the insertion operator. Notice that anything that
      60             : can be formatted with an insertion operator can go to the exception, even a \ref Vector
      61             : \verbatim
      62             :   Vector v;
      63             :   if(something_bad) throw Exception()<<"problem with this "<<v;
      64             : \endverbatim
      65             : In principle you can mix the two syntax (add a message as an argument and insert others with `<<`),
      66             : however it is not very clear and should be avoided.
      67             : We only allow using arguments in parenthesis in order to keep backward compatibility.
      68             : 
      69             : \par Using macros
      70             : 
      71             : In order to provide more context, especially for debugging, it might be useful to know where the exception
      72             : originated from. The macros below add information about the exact location of the error in the file (filename, line
      73             : and, when available, function name). Macros ending in "error" unconditionally throw
      74             : the exception, whereas macros ending in "assert" first perform a conditional check
      75             : (similarly to standard assert()).
      76             : An extra `m` in the name (e.g. `plumed_merror`) indicates a macro that provides a message as its argument.
      77             : However, as of PLUMED 2.5 we should prefer adding messages using insertion operators.
      78             : \verbatim
      79             : // this is correct but not recommended. add a message please!
      80             :   plumed_assert(a>0);
      81             : 
      82             : // this is the old syntax (with argument).
      83             : // this syntax is basically available for backward compatibility.
      84             :   plumed_massert(a>0,"a should be larger than zero);
      85             : 
      86             : // this is the recommended syntax, with insertion operators.
      87             : // it allows to easily insert multiple objects
      88             :   plumed_assert(a>0)<<"a should be larger than zero. a="<<a;
      89             : 
      90             : // same as above, but the test is made explicitly:
      91             :   if(a<=0) plumed_error();
      92             :   if(a<=0) plumed_error("a should be larger than zero);
      93             :   if(a<=0) plumed_error()<<"a should be larger than zero. a="<<a;
      94             : \endverbatim
      95             : 
      96             : The additional macros
      97             : plumed_dbg_assert() and plumed_dbg_massert() are similar
      98             : to plumed_assert() and plumed_massert() respectively, but the corresponding
      99             : check is only performed when NDEBUG macro is not defined. They should
     100             : be used when the check is expensive and should be skipped in production
     101             : code. So, for instance, in the following case:
     102             : \verbatim
     103             :   plumed_dbg_assert(expensive_function(i)>0)<<"message";
     104             : \endverbatim
     105             : `expensive_function()` is not called in the production code.
     106             : Notice that the compiler should be able to completely optimize away the
     107             : whole statement including functions used to produce the message as in this example:
     108             : \verbatim
     109             :   plumed_dbg_assert(expensive_function(i)>0)<<"I did this check "<<other_expensive_function(i);
     110             : \endverbatim
     111             : 
     112             : Finally, notice that there is another macro available, \ref plumed_here.
     113             : In can be used in order to create an exception with information about the
     114             : line/file coordinates without trowing it. That is, the two following syntaxes
     115             : are equivalent
     116             : \verbatim
     117             : // First way, all at once
     118             : plumed_error()<<"some message";
     119             : /////////////////////////////////
     120             : // Second way, one step at a time
     121             : // Create exception
     122             : Exception e;
     123             : // Append information about line and file
     124             : e<<plumed_here;
     125             : // Append some other message
     126             : e<<"some message";
     127             : // Throw the resulting exception
     128             : throw e;
     129             : \endverbatim
     130             : 
     131             : Exceptions can be caught within plumed or outside of it.
     132             : E.g., in an external c++ code using PLUMED as a library, one can type
     133             : \verbatim
     134             :   try{
     135             :     plumed.cmd("setPrecision",n);
     136             :   } catch (const std::exception & e) {
     137             :     std::printf("ee %s",e.what());
     138             :     exit(1);
     139             :   }
     140             : \endverbatim
     141             : This can be useful if an external code wants to exit in a controlled manner
     142             : (e.g. flushing files, printing the error message in a specific file, etc.)
     143             : but is anyway limited to c++ codes. Moreover,
     144             : since these errors are expected to be unrecoverable, the MD code will
     145             : usually not be able to do something more clever than exiting.
     146             : 
     147             : \note
     148             : We store message and stack trace in growing strings. This is in
     149             : principle not recommended, since copying the exception might fail if
     150             : copying the string throw another exception. However, this has been like
     151             : this in all previous PLUMED versions. In case it is necessary, we can replace
     152             : it later with a fixed size array placed on the stack.
     153             : 
     154             : */
     155             : class Exception : public std::exception {
     156             : /// Reported message. Can be updated.
     157             :   std::string msg;
     158             : /// Flag to remember if we have to write the `+++ message follows +++` string.
     159             : /// Needed so that the string appears only at the beginning of the message.
     160             :   bool note=true;
     161             : /// Stream used to insert objects.
     162             : /// It is not copied when the Exception is copied.
     163             :   std::stringstream stream;
     164             : /// Stack trace, computed at construction
     165             :   std::array<void*,128> callstack;
     166             : /// Number of frames in stack, computed at construction
     167             :   int callstack_n=0;
     168             : /// Parsed stack trace. Built at first use, thus mutable.
     169             :   mutable std::string stackTrace;
     170             : 
     171             : public:
     172             : 
     173             : /// Auxiliary containing the location of the exception in the file.
     174             : /// Typically used from the macros below.
     175             :   class Location {
     176             :   public:
     177             :     const char*file;
     178             :     const unsigned line;
     179             :     const char* pretty;
     180          54 :     explicit Location(const char*file,unsigned line,const char* pretty=nullptr):
     181          54 :       file(file),
     182          54 :       line(line),
     183          54 :       pretty(pretty)
     184             :     {}
     185             :   };
     186             : 
     187             : /// Auxiliary containing the failed assertion.
     188             : /// Typically used from the macros below.
     189             :   class Assertion {
     190             :   public:
     191             :     const char*assertion;
     192          10 :     explicit Assertion(const char*assertion=nullptr):
     193          10 :       assertion(assertion)
     194             :     {}
     195             :   };
     196             : 
     197             : /// Auxiliary class used to throw exceptions.
     198             : /// It just defines <<= operator so that:
     199             : /// - exceptions can be thrown calling std::throw_with_nested
     200             : ///   with a "throw like" syntax
     201             : /// - precedence is the same as the throw operator
     202             : /// (see https://en.cppreference.com/w/cpp/language/operator_precedence)
     203             :   class Throw {
     204             :   public:
     205             :     template <typename E>
     206           2 :     [[noreturn]] void operator <<=(E&&e) {
     207           4 :       if(std::current_exception()) {
     208           2 :         std::throw_with_nested(e);
     209             :       } else {
     210             :         // if not nested, avoid modifying the exception type
     211           0 :         throw e;
     212             :       }
     213             :     }
     214             :   };
     215             : 
     216             : /// Default constructor with no message.
     217             : /// Only records the stack trace.
     218             :   Exception();
     219             : 
     220             : /// Constructor compatible with PLUMED <=2.4.
     221          31 :   explicit Exception(const std::string & msg):
     222          31 :     Exception() {
     223          31 :     *this << msg;
     224          31 :   }
     225             : 
     226             : /// Copy constructor.
     227             : /// Needed to make sure stream is not copied
     228          70 :   Exception(const Exception & e):
     229          70 :     msg(e.msg),
     230          70 :     note(e.note),
     231          70 :     callstack(e.callstack),
     232          70 :     callstack_n(e.callstack_n),
     233          70 :     stackTrace(e.stackTrace) {
     234          70 :   }
     235             : 
     236             : /// Assignment.
     237             : /// Needed to make sure stream is not copied
     238             :   Exception & operator=(const Exception & e) {
     239             :     msg=e.msg;
     240             :     note=e.note;
     241             :     callstack=e.callstack;
     242             :     callstack_n=e.callstack_n;
     243             :     stackTrace=e.stackTrace;
     244             :     stream.str("");
     245             :     return *this;
     246             :   }
     247             : 
     248             : /// Returns the error message.
     249             : /// In case the environment variable PLUMED_STACK_TRACE was defined
     250             : /// and equal to `yes` when the exception was raised,
     251             : /// the error message will contain the stack trace as well.
     252         121 :   const char* what() const noexcept override {
     253         121 :     return msg.c_str();
     254             :   }
     255             : 
     256             : /// Returns the stack trace as a string.
     257             : /// This function is slow as it requires building a parsed string.
     258             : /// If storing the stack for later usage, you might prefer to use trace().
     259             :   const char* stack() const;
     260             : 
     261             : /// Returns the callstack.
     262             :   const std::array<void*,128> & trace() const noexcept {
     263             :     return callstack;
     264             :   }
     265             : 
     266             : /// Returns the number of elements in the trace array
     267             :   int trace_n() const noexcept {
     268             :     return callstack_n;
     269             :   }
     270             : 
     271             : /// Destructor should be defined and should not throw other exceptions
     272         144 :   ~Exception() noexcept override {}
     273             : 
     274             : /// Insert location.
     275             : /// Format the location properly.
     276             :   Exception& operator<<(const Location&);
     277             : 
     278             : /// Insert assertion.
     279             : /// Format the assertion properly
     280             :   Exception& operator<<(const Assertion&);
     281             : 
     282             : /// Insert string.
     283             : /// Append this string to the message.
     284             :   Exception& operator<<(const std::string&);
     285             : 
     286             : /// Insert anything else.
     287             : /// This allows to dump also other types (e.g. double, or even Vector).
     288             : /// Anything that can be written on a stream can go here.
     289             :   template<typename T>
     290         267 :   Exception& operator<<(const T & x) {
     291         267 :     stream<<x;
     292         267 :     (*this)<<stream.str();
     293         267 :     stream.str("");
     294         267 :     return *this;
     295             :   }
     296             : };
     297             : 
     298             : /// Class representing a generic error
     299         143 : class ExceptionError :
     300             :   public Exception {
     301             : public:
     302           1 :   using Exception::Exception;
     303             :   template<typename T>
     304             :   ExceptionError& operator<<(const T & x) {
     305          89 :     *static_cast<Exception*>(this) <<x;
     306             :     return *this;
     307             :   }
     308             : };
     309             : 
     310             : /// Class representing a debug error (can only be thrown when using debug options)
     311           0 : class ExceptionDebug :
     312             :   public Exception {
     313             : public:
     314           1 :   using Exception::Exception;
     315             :   template<typename T>
     316             :   ExceptionDebug& operator<<(const T & x) {
     317             :     *static_cast<Exception*>(this) <<x;
     318             :     return *this;
     319             :   }
     320             : };
     321             : 
     322             : /// Class representing a type error in the PLMD::Plumed interface
     323         111 : class ExceptionTypeError :
     324             :   public Exception {
     325             : public:
     326           0 :   using Exception::Exception;
     327             :   template<typename T>
     328             :   ExceptionTypeError& operator<<(const T & x) {
     329         107 :     *static_cast<Exception*>(this) <<x;
     330             :     return *this;
     331             :   }
     332             : };
     333             : 
     334             : #ifdef __GNUG__
     335             : // With GNU compiler, we can use __PRETTY_FUNCTION__ to get the function name
     336             : #define __PLUMED_FUNCNAME __PRETTY_FUNCTION__
     337             : #else
     338             : // Otherwise, we use the standard C++11 variable
     339             : #define __PLUMED_FUNCNAME __func__
     340             : #endif
     341             : 
     342             : #ifndef PLUMED_MODULE_DIR
     343             : #define PLUMED_MODULE_DIR ""
     344             : #endif
     345             : 
     346             : /// \relates PLMD::Exception
     347             : /// Auxiliary macro that generates a PLMD::Exception::Location object.
     348             : /// Might be useful if we want to use derived exceptions that could
     349             : /// be thrown using `throw DerivedException()<<plumed_here<<" "<<other stuff"`.
     350             : /// It is used in the macros below to throw PLMD::Exception.
     351             : #define plumed_here PLMD::Exception::Location(PLUMED_MODULE_DIR __FILE__,__LINE__,__PLUMED_FUNCNAME)
     352             : 
     353             : /// \relates PLMD::Exception
     354             : /// Throw an exception with information about the position in the file.
     355             : /// Messages can be inserted with `plumed_error()<<"message"`.
     356             : #define plumed_error() throw PLMD::ExceptionError() << plumed_here
     357             : 
     358             : /// \relates PLMD::Exception
     359             : /// Throw a nested exception with information about the position in the file.
     360             : /// It preliminary checks if we are in a catch block. If so, the caught exception
     361             : /// is rethrown as nested. If not, it throws a normal ExceptionError.
     362             : /// NB in theory we could have just redefined plumed_error to this, but
     363             : /// for some reason cppcheck does not understand that the <<= operator used here is
     364             : /// [[noreturn]] and thus gives many false warnings
     365             : #define plumed_error_nested() PLMD::Exception::Throw() <<= PLMD::ExceptionError() << plumed_here
     366             : 
     367             : /// \relates PLMD::Exception
     368             : /// Throw an exception with information about the position in the file
     369             : /// and a message. Mostly available for backward compatibility
     370             : #define plumed_merror(msg) plumed_error() << msg
     371             : 
     372             : /// \relates PLMD::Exception
     373             : /// Launches plumed_merror only if test evaluates to false.
     374             : /// The string describing the test is also reported.
     375             : /// Further messages can be inserted with `<<`.
     376             : #define plumed_assert(test) if(!(test)) plumed_error() << PLMD::Exception::Assertion(#test)
     377             : 
     378             : /// \relates PLMD::Exception
     379             : /// Launches plumed_merror only if test evaluates to false.
     380             : /// The string describing the test is also reported, in addition to
     381             : /// messages reported in the extra argument. Mostly available for backward compatibility.
     382             : #define plumed_massert(test,msg) plumed_assert(test) << msg
     383             : 
     384             : #ifdef NDEBUG
     385             : 
     386             : // These are the versions used when compiling with NDEBUG flag.
     387             : // The if constexpr(false) gurarantees that the compiler will optimize away the assertion
     388             : // We are not using an empty macro becasue the user may want to use the << operator
     389             : 
     390             : #define plumed_dbg_assert(test) if constexpr(false) plumed_assert(true)
     391             : #define plumed_dbg_massert(test,msg) if constexpr(false) plumed_massert(true,msg)
     392             : 
     393             : #else
     394             : 
     395             : /// \relates PLMD::Exception
     396             : /// Same as \ref plumed_assert, but only evaluates the condition if NDEBUG is not defined.
     397             : #define plumed_dbg_assert(test) if(!(test)) PLMD::Exception::Throw() <<= PLMD::ExceptionDebug() << plumed_here << PLMD::Exception::Assertion(#test) << "(this check is enabled only in debug builds)\n"
     398             : 
     399             : /// \relates PLMD::Exception
     400             : /// Same as \ref plumed_massert, but only evaluates the condition if NDEBUG is not defined.
     401             : #define plumed_dbg_massert(test,msg) plumed_dbg_assert(test) << msg
     402             : 
     403             : #endif
     404             : 
     405             : }
     406             : #endif

Generated by: LCOV version 1.16