#include <Exception.h>
Classes | |
class | Assertion |
Auxiliary containing the failed assertion. More... | |
class | Location |
Auxiliary containing the location of the exception in the file. More... | |
Public Member Functions | |
Exception () | |
Default constructor with no message. More... | |
Exception (const Exception &e) | |
Copy constructor. More... | |
Exception (const std::string &msg) | |
Constructor compatible with PLUMED <=2.4. More... | |
~Exception () noexcept override | |
Destructor should be defined and should not throw other exceptions. More... | |
Exception & | operator<< (const Assertion &) |
Insert assertion. More... | |
Exception & | operator<< (const Location &) |
Insert location. More... | |
Exception & | operator<< (const std::string &) |
Insert string. More... | |
template<typename T > | |
Exception & | operator<< (const T &x) |
Insert anything else. More... | |
Exception & | operator= (const Exception &e) |
Assignment. More... | |
const char * | stack () const |
Returns the stack trace as a string. More... | |
const std::array< void *, 128 > & | trace () const noexcept |
Returns the callstack. More... | |
int | trace_n () const noexcept |
Returns the number of elements in the trace array. More... | |
const char * | what () const noexcept override |
Returns the error message. More... | |
Private Attributes | |
std::array< void *, 128 > | callstack |
Stack trace, computed at construction. More... | |
int | callstack_n =0 |
Number of frames in stack, computed at construction. More... | |
std::string | msg |
Reported message. Can be updated. More... | |
bool | note =true |
Flag to remember if we have to write the +++ message follows +++ string. More... | |
std::string | stackTrace |
Parsed stack trace. Built at first use, thus mutable. More... | |
std::stringstream | stream |
Stream used to insert objects. More... | |
Related Functions | |
(Note that these are not member functions.) | |
#define | plumed_assert(test) if(!(test)) plumed_error() << PLMD::Exception::Assertion(#test) |
#define | plumed_dbg_assert(test) if(!(test)) throw PLMD::ExceptionDebug() << plumed_here << PLMD::Exception::Assertion(#test) << "(this check is enabled only in debug builds)\n" |
#define | plumed_dbg_massert(test, msg) plumed_dbg_assert(test) << msg |
#define | plumed_error() throw PLMD::ExceptionError() << plumed_here |
#define | plumed_here PLMD::Exception::Location(__FILE__,__LINE__,__PLUMED_FUNCNAME) |
#define | plumed_massert(test, msg) plumed_assert(test) << msg |
#define | plumed_merror(msg) plumed_error() << msg |
Class to deal with Plumed runtime errors.
This class and the related macros can be used to detect programming errors. Typical cases are internal inconsistencies or errors in the plumed<->MD interface. Mistakes made by final users (i.e. in the plumed.dat
file) should probably be documented in some better way (e.g. printing parts of the manual in the output). However, also this class allows for significant information to be attached. Let's try to make error messages as informative as possible!
plumed_error()<<"check this vector "<<v;
See below for more details.To throw an error, just throw a c++ exception
if(something_bad) throw Exception();
or better add an error message to that
if(something_bad) throw Exception("describe the error here");
As of PLUMED 2.5 you can add multiple messages, they will just be concatenated, but to do se you should use the insertion operator. Notice that anything that can be formatted with an insertion operator can go to the exception, even a Vector
Vector v; if(something_bad) throw Exception()<<"problem with this "<<v;
In principle you can mix the two syntax (add a message as an argument and insert others with <<
), however it is not very clear and should be avoided. We only allow using arguments in parenthesis in order to keep backward compatibility.
In order to provide more context, especially for debugging, it might be useful to know where the exception originated from. The macros below add information about the exact location of the error in the file (filename, line and, when available, function name). Macros ending in "error" unconditionally throw the exception, whereas macros ending in "assert" first perform a conditional check (similarly to standard assert()). An extra m
in the name (e.g. plumed_merror
) indicates a macro that provides a message as its argument. However, as of PLUMED 2.5 we should prefer adding messages using insertion operators.
// this is correct but not recommended. add a message please! plumed_assert(a>0); // this is the old syntax (with argument). // this syntax is basically available for backward compatibility. plumed_massert(a>0,"a should be larger than zero); // this is the recommended syntax, with insertion operators. // it allows to easily insert multiple objects plumed_assert(a>0)<<"a should be larger than zero. a="<<a; // same as above, but the test is made explicitly: if(a<=0) plumed_error(); if(a<=0) plumed_error("a should be larger than zero); if(a<=0) plumed_error()<<"a should be larger than zero. a="<<a;
The additional macros plumed_dbg_assert() and plumed_dbg_massert() are similar to plumed_assert() and plumed_massert() respectively, but the corresponding check is only performed when NDEBUG macro is not defined. They should be used when the check is expensive and should be skipped in production code. So, for instance, in the following case:
plumed_dbg_assert(expensive_function(i)>0)<<"message";
expensive_function()
is not called in the production code. Notice that the compiler should be able to completely optimize away the whole statement including functions used to produce the message as in this example:
plumed_dbg_assert(expensive_function(i)>0)<<"I did this check "<<other_expensive_function(i);
Finally, notice that there is another macro available, plumed_here. In can be used in order to create an exception with information about the line/file coordinates without trowing it. That is, the two following syntaxes are equivalent
// First way, all at once plumed_error()<<"some message"; ///////////////////////////////// // Second way, one step at a time // Create exception Exception e; // Append information about line and file e<<plumed_here; // Append some other message e<<"some message"; // Throw the resulting exception throw e;
Exceptions can be caught within plumed or outside of it. E.g., in an external c++ code using PLUMED as a library, one can type
try{ plumed.cmd("setPrecision",n); } catch (const std::exception & e) { std::printf("ee %s",e.what()); exit(1); }
This can be useful if an external code wants to exit in a controlled manner (e.g. flushing files, printing the error message in a specific file, etc.) but is anyway limited to c++ codes. Moreover, since these errors are expected to be unrecoverable, the MD code will usually not be able to do something more clever than exiting.
PLMD::Exception::Exception | ( | ) |
Default constructor with no message.
Only records the stack trace.
|
inlineexplicit |
Constructor compatible with PLUMED <=2.4.
|
inline |
Copy constructor.
Needed to make sure stream is not copied
|
inlineoverridenoexcept |
Destructor should be defined and should not throw other exceptions.
Insert assertion.
Format the assertion properly
Insert location.
Format the location properly.
Exception& PLMD::Exception::operator<< | ( | const std::string & | ) |
Insert string.
Append this string to the message.
|
inline |
Insert anything else.
This allows to dump also other types (e.g. double, or even Vector). Anything that can be written on a stream can go here.
Assignment.
Needed to make sure stream is not copied
const char * PLMD::Exception::stack | ( | ) | const |
Returns the stack trace as a string.
This function is slow as it requires building a parsed string. If storing the stack for later usage, you might prefer to use trace().
|
inlinenoexcept |
Returns the callstack.
|
inlinenoexcept |
Returns the number of elements in the trace array.
|
inlineoverridenoexcept |
Returns the error message.
In case the environment variable PLUMED_STACK_TRACE was defined and equal to yes
when the exception was raised, the error message will contain the stack trace as well.
|
related |
Launches plumed_merror only if test evaluates to false. The string describing the test is also reported. Further messages can be inserted with <<
.
|
related |
Same as plumed_assert, but only evaluates the condition if NDEBUG is not defined.
|
related |
Same as plumed_massert, but only evaluates the condition if NDEBUG is not defined.
|
related |
Throw an exception with information about the position in the file. Messages can be inserted with plumed_error()<<"message"
.
|
related |
Auxiliary macro that generates a PLMD::Exception::Location object. Might be useful if we want to use derived exceptions that could be thrown using throw DerivedException()<<plumed_here<<" "<<other stuff"
. It is used in the macros below to throw PLMD::Exception.
|
related |
Launches plumed_merror only if test evaluates to false. The string describing the test is also reported, in addition to messages reported in the extra argument. Mostly available for backward compatibility.
|
related |
Throw an exception with information about the position in the file and a message. Mostly available for backward compatibility
|
private |
Stack trace, computed at construction.
|
private |
Number of frames in stack, computed at construction.
|
private |
Reported message. Can be updated.
|
private |
Flag to remember if we have to write the +++ message follows +++
string.
Needed so that the string appears only at the beginning of the message.
|
mutableprivate |
Parsed stack trace. Built at first use, thus mutable.
|
private |
Stream used to insert objects.
It is not copied when the Exception is copied.
Hosted by GitHub | 1.8.17 |